diff src/w32fns.c @ 13434:53ba95a88cf2

Initial revision
author Geoff Voelker <voelker@cs.washington.edu>
date Tue, 07 Nov 1995 07:52:28 +0000
parents
children ee40177f6c68
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/w32fns.c	Tue Nov 07 07:52:28 1995 +0000
@@ -0,0 +1,4308 @@
+/* Functions for the Win32 window system.
+   Copyright (C) 1989, 1992, 1993, 1994, 1995 Free Software Foundation.
+
+This file is part of GNU Emacs.
+
+GNU Emacs 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.
+
+GNU Emacs 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 GNU Emacs; see the file COPYING.  If not, write to
+the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.  */
+
+/* Added by Kevin Gallo */
+
+#include <signal.h>
+#include <config.h>
+#include <stdio.h>
+
+#include "lisp.h"
+#include "w32term.h"
+#include "frame.h"
+#include "window.h"
+#include "buffer.h"
+#include "dispextern.h"
+#include "keyboard.h"
+#include "blockinput.h"
+#include "paths.h"
+#include "ntheap.h"
+#include "termhooks.h"
+
+#include <commdlg.h>
+
+extern void abort ();
+extern void free_frame_menubar ();
+extern struct scroll_bar *x_window_to_scroll_bar ();
+
+/* The colormap for converting color names to RGB values */
+Lisp_Object Vwin32_color_map;
+
+/* The name we're using in resource queries.  */
+Lisp_Object Vx_resource_name;
+
+/* Non nil if no window manager is in use.  */
+Lisp_Object Vx_no_window_manager;
+
+/* The background and shape of the mouse pointer, and shape when not
+   over text or in the modeline.  */
+Lisp_Object Vx_pointer_shape, Vx_nontext_pointer_shape, Vx_mode_pointer_shape;
+/* The shape when over mouse-sensitive text.  */
+Lisp_Object Vx_sensitive_text_pointer_shape;
+
+/* Color of chars displayed in cursor box.  */
+Lisp_Object Vx_cursor_fore_pixel;
+
+/* Search path for bitmap files.  */
+Lisp_Object Vx_bitmap_file_path;
+
+/* Evaluate this expression to rebuild the section of syms_of_w32fns
+   that initializes and staticpros the symbols declared below.  Note
+   that Emacs 18 has a bug that keeps C-x C-e from being able to
+   evaluate this expression.
+
+(progn
+  ;; Accumulate a list of the symbols we want to initialize from the
+  ;; declarations at the top of the file.
+  (goto-char (point-min))
+  (search-forward "/\*&&& symbols declared here &&&*\/\n")
+  (let (symbol-list)
+    (while (looking-at "Lisp_Object \\(Q[a-z_]+\\)")
+      (setq symbol-list
+	    (cons (buffer-substring (match-beginning 1) (match-end 1))
+		  symbol-list))
+      (forward-line 1))
+    (setq symbol-list (nreverse symbol-list))
+    ;; Delete the section of syms_of_... where we initialize the symbols.
+    (search-forward "\n  /\*&&& init symbols here &&&*\/\n")
+    (let ((start (point)))
+      (while (looking-at "^  Q")
+	(forward-line 2))
+      (kill-region start (point)))
+    ;; Write a new symbol initialization section.
+    (while symbol-list
+      (insert (format "  %s = intern (\"" (car symbol-list)))
+      (let ((start (point)))
+	(insert (substring (car symbol-list) 1))
+	(subst-char-in-region start (point) ?_ ?-))
+      (insert (format "\");\n  staticpro (&%s);\n" (car symbol-list)))
+      (setq symbol-list (cdr symbol-list)))))
+
+  */        
+
+/*&&& symbols declared here &&&*/
+Lisp_Object Qauto_raise;
+Lisp_Object Qauto_lower;
+Lisp_Object Qbackground_color;
+Lisp_Object Qbar;
+Lisp_Object Qborder_color;
+Lisp_Object Qborder_width;
+Lisp_Object Qbox;
+Lisp_Object Qcursor_color;
+Lisp_Object Qcursor_type;
+Lisp_Object Qfont;
+Lisp_Object Qforeground_color;
+Lisp_Object Qgeometry;
+Lisp_Object Qicon_left;
+Lisp_Object Qicon_top;
+Lisp_Object Qicon_type;
+Lisp_Object Qicon_name;
+Lisp_Object Qinternal_border_width;
+Lisp_Object Qleft;
+Lisp_Object Qmouse_color;
+Lisp_Object Qnone;
+Lisp_Object Qparent_id;
+Lisp_Object Qscroll_bar_width;
+Lisp_Object Qsuppress_icon;
+Lisp_Object Qtop;
+Lisp_Object Qundefined_color;
+Lisp_Object Qvertical_scroll_bars;
+Lisp_Object Qvisibility;
+Lisp_Object Qwindow_id;
+Lisp_Object Qx_frame_parameter;
+Lisp_Object Qx_resource_name;
+Lisp_Object Quser_position;
+Lisp_Object Quser_size;
+Lisp_Object Qdisplay;
+
+/* The below are defined in frame.c.  */
+extern Lisp_Object Qheight, Qminibuffer, Qname, Qonly, Qwidth;
+extern Lisp_Object Qunsplittable, Qmenu_bar_lines;
+
+extern Lisp_Object Vwindow_system_version;
+
+extern Lisp_Object last_mouse_scroll_bar;
+extern int last_mouse_scroll_bar_pos;
+Time last_mouse_movement_time;
+
+
+/* Extract a frame as a FRAME_PTR, defaulting to the selected frame
+   and checking validity for Win32.  */
+
+FRAME_PTR
+check_x_frame (frame)
+     Lisp_Object frame;
+{
+  FRAME_PTR f;
+
+  if (NILP (frame))
+    f = selected_frame;
+  else
+    {
+      CHECK_LIVE_FRAME (frame, 0);
+      f = XFRAME (frame);
+    }
+  if (! FRAME_WIN32_P (f))
+    error ("non-win32 frame used");
+  return f;
+}
+
+/* Let the user specify an display with a frame.
+   nil stands for the selected frame--or, if that is not a win32 frame,
+   the first display on the list.  */
+
+static struct win32_display_info *
+check_x_display_info (frame)
+     Lisp_Object frame;
+{
+  if (NILP (frame))
+    {
+      if (FRAME_WIN32_P (selected_frame))
+	return FRAME_WIN32_DISPLAY_INFO (selected_frame);
+      else
+	return &one_win32_display_info;
+    }
+  else if (STRINGP (frame))
+    return x_display_info_for_name (frame);
+  else
+    {
+      FRAME_PTR f;
+
+      CHECK_LIVE_FRAME (frame, 0);
+      f = XFRAME (frame);
+      if (! FRAME_WIN32_P (f))
+	error ("non-win32 frame used");
+      return FRAME_WIN32_DISPLAY_INFO (f);
+    }
+}
+
+/* Return the Emacs frame-object corresponding to an win32 window.
+   It could be the frame's main window or an icon window.  */
+
+/* This function can be called during GC, so use GC_xxx type test macros.  */
+
+struct frame *
+x_window_to_frame (dpyinfo, wdesc)
+     struct win32_display_info *dpyinfo;
+     HWND wdesc;
+{
+  Lisp_Object tail, frame;
+  struct frame *f;
+
+  for (tail = Vframe_list; GC_CONSP (tail); tail = XCONS (tail)->cdr)
+    {
+      frame = XCONS (tail)->car;
+      if (!GC_FRAMEP (frame))
+        continue;
+      f = XFRAME (frame);
+      if (f->output_data.nothing == 1 
+	  || FRAME_WIN32_DISPLAY_INFO (f) != dpyinfo)
+	continue;
+      if (FRAME_WIN32_WINDOW (f) == wdesc)
+        return f;
+    }
+  return 0;
+}
+
+
+
+/* Code to deal with bitmaps.  Bitmaps are referenced by their bitmap
+   id, which is just an int that this section returns.  Bitmaps are
+   reference counted so they can be shared among frames.
+
+   Bitmap indices are guaranteed to be > 0, so a negative number can
+   be used to indicate no bitmap.
+
+   If you use x_create_bitmap_from_data, then you must keep track of
+   the bitmaps yourself.  That is, creating a bitmap from the same
+   data more than once will not be caught.  */
+
+
+/* Functions to access the contents of a bitmap, given an id.  */
+
+int
+x_bitmap_height (f, id)
+     FRAME_PTR f;
+     int id;
+{
+  return FRAME_WIN32_DISPLAY_INFO (f)->bitmaps[id - 1].height;
+}
+
+int
+x_bitmap_width (f, id)
+     FRAME_PTR f;
+     int id;
+{
+  return FRAME_WIN32_DISPLAY_INFO (f)->bitmaps[id - 1].width;
+}
+
+int
+x_bitmap_pixmap (f, id)
+     FRAME_PTR f;
+     int id;
+{
+  return (int) FRAME_WIN32_DISPLAY_INFO (f)->bitmaps[id - 1].pixmap;
+}
+
+
+/* Allocate a new bitmap record.  Returns index of new record.  */
+
+static int
+x_allocate_bitmap_record (f)
+     FRAME_PTR f;
+{
+  struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (f);
+  int i;
+
+  if (dpyinfo->bitmaps == NULL)
+    {
+      dpyinfo->bitmaps_size = 10;
+      dpyinfo->bitmaps
+	= (struct win32_bitmap_record *) xmalloc (dpyinfo->bitmaps_size * sizeof (struct win32_bitmap_record));
+      dpyinfo->bitmaps_last = 1;
+      return 1;
+    }
+
+  if (dpyinfo->bitmaps_last < dpyinfo->bitmaps_size)
+    return ++dpyinfo->bitmaps_last;
+
+  for (i = 0; i < dpyinfo->bitmaps_size; ++i)
+    if (dpyinfo->bitmaps[i].refcount == 0)
+      return i + 1;
+
+  dpyinfo->bitmaps_size *= 2;
+  dpyinfo->bitmaps
+    = (struct win32_bitmap_record *) xrealloc (dpyinfo->bitmaps,
+					   dpyinfo->bitmaps_size * sizeof (struct win32_bitmap_record));
+  return ++dpyinfo->bitmaps_last;
+}
+
+/* Add one reference to the reference count of the bitmap with id ID.  */
+
+void
+x_reference_bitmap (f, id)
+     FRAME_PTR f;
+     int id;
+{
+  ++FRAME_WIN32_DISPLAY_INFO (f)->bitmaps[id - 1].refcount;
+}
+
+/* Create a bitmap for frame F from a HEIGHT x WIDTH array of bits at BITS.  */
+
+int
+x_create_bitmap_from_data (f, bits, width, height)
+     struct frame *f;
+     char *bits;
+     unsigned int width, height;
+{
+  struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (f);
+  Pixmap bitmap;
+  int id;
+
+  bitmap = CreateBitmap (width, height,
+			 FRAME_WIN32_DISPLAY_INFO (XFRAME (frame))->n_planes,
+			 FRAME_WIN32_DISPLAY_INFO (XFRAME (frame))->n_cbits,
+			 bits);
+
+  if (! bitmap)
+    return -1;
+
+  id = x_allocate_bitmap_record (f);
+  dpyinfo->bitmaps[id - 1].pixmap = bitmap;
+  dpyinfo->bitmaps[id - 1].file = NULL;
+  dpyinfo->bitmaps[id - 1].hinst = NULL;
+  dpyinfo->bitmaps[id - 1].refcount = 1;
+  dpyinfo->bitmaps[id - 1].depth = 1;
+  dpyinfo->bitmaps[id - 1].height = height;
+  dpyinfo->bitmaps[id - 1].width = width;
+
+  return id;
+}
+
+/* Create bitmap from file FILE for frame F.  */
+
+int
+x_create_bitmap_from_file (f, file)
+     struct frame *f;
+     Lisp_Object file;
+{
+  return -1;
+#if 0
+  struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (f);
+  unsigned int width, height;
+  Pixmap bitmap;
+  int xhot, yhot, result, id;
+  Lisp_Object found;
+  int fd;
+  char *filename;
+  HINSTANCE hinst;
+
+  /* Look for an existing bitmap with the same name.  */
+  for (id = 0; id < dpyinfo->bitmaps_last; ++id)
+    {
+      if (dpyinfo->bitmaps[id].refcount
+	  && dpyinfo->bitmaps[id].file
+	  && !strcmp (dpyinfo->bitmaps[id].file, (char *) XSTRING (file)->data))
+	{
+	  ++dpyinfo->bitmaps[id].refcount;
+	  return id + 1;
+	}
+    }
+
+  /* Search bitmap-file-path for the file, if appropriate.  */
+  fd = openp (Vx_bitmap_file_path, file, "", &found, 0);
+  if (fd < 0)
+    return -1;
+  close (fd);
+
+  filename = (char *) XSTRING (found)->data;
+
+  hinst = LoadLibraryEx (filename, NULL, LOAD_LIBRARY_AS_DATAFILE);
+
+  if (hinst == NULL)
+      return -1;
+
+  
+  result = XReadBitmapFile (FRAME_WIN32_DISPLAY (f), FRAME_WIN32_WINDOW (f),
+			    filename, &width, &height, &bitmap, &xhot, &yhot);
+  if (result != BitmapSuccess)
+    return -1;
+
+  id = x_allocate_bitmap_record (f);
+  dpyinfo->bitmaps[id - 1].pixmap = bitmap;
+  dpyinfo->bitmaps[id - 1].refcount = 1;
+  dpyinfo->bitmaps[id - 1].file = (char *) xmalloc (XSTRING (file)->size + 1);
+  dpyinfo->bitmaps[id - 1].depth = 1;
+  dpyinfo->bitmaps[id - 1].height = height;
+  dpyinfo->bitmaps[id - 1].width = width;
+  strcpy (dpyinfo->bitmaps[id - 1].file, XSTRING (file)->data);
+
+  return id;
+#endif
+}
+
+/* Remove reference to bitmap with id number ID.  */
+
+int
+x_destroy_bitmap (f, id)
+     FRAME_PTR f;
+     int id;
+{
+  struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (f);
+
+  if (id > 0)
+    {
+      --dpyinfo->bitmaps[id - 1].refcount;
+      if (dpyinfo->bitmaps[id - 1].refcount == 0)
+	{
+	  BLOCK_INPUT;
+	  DeleteObject (dpyinfo->bitmaps[id - 1].pixmap);
+	  if (dpyinfo->bitmaps[id - 1].file)
+	    {
+	      free (dpyinfo->bitmaps[id - 1].file);
+	      dpyinfo->bitmaps[id - 1].file = NULL;
+	    }
+	  UNBLOCK_INPUT;
+	}
+    }
+}
+
+/* Free all the bitmaps for the display specified by DPYINFO.  */
+
+static void
+x_destroy_all_bitmaps (dpyinfo)
+     struct win32_display_info *dpyinfo;
+{
+  int i;
+  for (i = 0; i < dpyinfo->bitmaps_last; i++)
+    if (dpyinfo->bitmaps[i].refcount > 0)
+      {
+	DeleteObject (dpyinfo->bitmaps[i].pixmap);
+	if (dpyinfo->bitmaps[i].file)
+	  free (dpyinfo->bitmaps[i].file);
+      }
+  dpyinfo->bitmaps_last = 0;
+}
+
+/* Connect the frame-parameter names for Win32 frames
+   to the ways of passing the parameter values to the window system.
+
+   The name of a parameter, as a Lisp symbol,
+   has an `x-frame-parameter' property which is an integer in Lisp
+   but can be interpreted as an `enum x_frame_parm' in C.  */
+
+enum x_frame_parm
+{
+  X_PARM_FOREGROUND_COLOR,
+  X_PARM_BACKGROUND_COLOR,
+  X_PARM_MOUSE_COLOR,
+  X_PARM_CURSOR_COLOR,
+  X_PARM_BORDER_COLOR,
+  X_PARM_ICON_TYPE,
+  X_PARM_FONT,
+  X_PARM_BORDER_WIDTH,
+  X_PARM_INTERNAL_BORDER_WIDTH,
+  X_PARM_NAME,
+  X_PARM_AUTORAISE,
+  X_PARM_AUTOLOWER,
+  X_PARM_VERT_SCROLL_BAR,
+  X_PARM_VISIBILITY,
+  X_PARM_MENU_BAR_LINES
+};
+
+
+struct x_frame_parm_table
+{
+  char *name;
+  void (*setter)( /* struct frame *frame, Lisp_Object val, oldval */ );
+};
+
+void x_set_foreground_color ();
+void x_set_background_color ();
+void x_set_mouse_color ();
+void x_set_cursor_color ();
+void x_set_border_color ();
+void x_set_cursor_type ();
+void x_set_icon_type ();
+void x_set_icon_name ();
+void x_set_font ();
+void x_set_border_width ();
+void x_set_internal_border_width ();
+void x_explicitly_set_name ();
+void x_set_autoraise ();
+void x_set_autolower ();
+void x_set_vertical_scroll_bars ();
+void x_set_visibility ();
+void x_set_menu_bar_lines ();
+void x_set_scroll_bar_width ();
+void x_set_unsplittable ();
+
+static struct x_frame_parm_table x_frame_parms[] =
+{
+  "foreground-color", x_set_foreground_color,
+  "background-color", x_set_background_color,
+  "mouse-color", x_set_mouse_color,
+  "cursor-color", x_set_cursor_color,
+  "border-color", x_set_border_color,
+  "cursor-type", x_set_cursor_type,
+  "icon-type", x_set_icon_type,
+  "icon-name", x_set_icon_name,
+  "font", x_set_font,
+  "border-width", x_set_border_width,
+  "internal-border-width", x_set_internal_border_width,
+  "name", x_explicitly_set_name,
+  "auto-raise", x_set_autoraise,
+  "auto-lower", x_set_autolower,
+  "vertical-scroll-bars", x_set_vertical_scroll_bars,
+  "visibility", x_set_visibility,
+  "menu-bar-lines", x_set_menu_bar_lines,
+  "scroll-bar-width", x_set_scroll_bar_width,
+  "unsplittable", x_set_unsplittable,
+};
+
+/* Attach the `x-frame-parameter' properties to
+   the Lisp symbol names of parameters relevant to Win32.  */
+
+init_x_parm_symbols ()
+{
+  int i;
+
+  for (i = 0; i < sizeof (x_frame_parms) / sizeof (x_frame_parms[0]); i++)
+    Fput (intern (x_frame_parms[i].name), Qx_frame_parameter,
+	  make_number (i));
+}
+
+/* Change the parameters of FRAME as specified by ALIST.
+   If a parameter is not specially recognized, do nothing;
+   otherwise call the `x_set_...' function for that parameter.  */
+
+void
+x_set_frame_parameters (f, alist)
+     FRAME_PTR f;
+     Lisp_Object alist;
+{
+  Lisp_Object tail;
+
+  /* If both of these parameters are present, it's more efficient to
+     set them both at once.  So we wait until we've looked at the
+     entire list before we set them.  */
+  Lisp_Object width, height;
+
+  /* Same here.  */
+  Lisp_Object left, top;
+
+  /* Same with these.  */
+  Lisp_Object icon_left, icon_top;
+
+  /* Record in these vectors all the parms specified.  */
+  Lisp_Object *parms;
+  Lisp_Object *values;
+  int i;
+  int left_no_change = 0, top_no_change = 0;
+  int icon_left_no_change = 0, icon_top_no_change = 0;
+
+  i = 0;
+  for (tail = alist; CONSP (tail); tail = Fcdr (tail))
+    i++;
+
+  parms = (Lisp_Object *) alloca (i * sizeof (Lisp_Object));
+  values = (Lisp_Object *) alloca (i * sizeof (Lisp_Object));
+
+  /* Extract parm names and values into those vectors.  */
+
+  i = 0;
+  for (tail = alist; CONSP (tail); tail = Fcdr (tail))
+    {
+      Lisp_Object elt, prop, val;
+
+      elt = Fcar (tail);
+      parms[i] = Fcar (elt);
+      values[i] = Fcdr (elt);
+      i++;
+    }
+
+  width = height = top = left = Qunbound;
+  icon_left = icon_top = Qunbound;
+
+  /* Now process them in reverse of specified order.  */
+  for (i--; i >= 0; i--)
+    {
+      Lisp_Object prop, val;
+
+      prop = parms[i];
+      val = values[i];
+
+      if (EQ (prop, Qwidth))
+	width = val;
+      else if (EQ (prop, Qheight))
+	height = val;
+      else if (EQ (prop, Qtop))
+	top = val;
+      else if (EQ (prop, Qleft))
+	left = val;
+      else if (EQ (prop, Qicon_top))
+	icon_top = val;
+      else if (EQ (prop, Qicon_left))
+	icon_left = val;
+      else
+	{
+	  register Lisp_Object param_index, old_value;
+
+	  param_index = Fget (prop, Qx_frame_parameter);
+	  old_value = get_frame_param (f, prop);
+	  store_frame_param (f, prop, val);
+ 	  if (NATNUMP (param_index)
+	      && (XFASTINT (param_index)
+		  < sizeof (x_frame_parms)/sizeof (x_frame_parms[0])))
+	    (*x_frame_parms[XINT (param_index)].setter)(f, val, old_value);
+	}
+    }
+
+  /* Don't die if just one of these was set.  */
+  if (EQ (left, Qunbound))
+    {
+      left_no_change = 1;
+      if (f->output_data.win32->left_pos < 0)
+	left = Fcons (Qplus, Fcons (make_number (f->output_data.win32->left_pos), Qnil));
+      else
+	XSETINT (left, f->output_data.win32->left_pos);
+    }
+  if (EQ (top, Qunbound))
+    {
+      top_no_change = 1;
+      if (f->output_data.win32->top_pos < 0)
+	top = Fcons (Qplus, Fcons (make_number (f->output_data.win32->top_pos), Qnil));
+      else
+	XSETINT (top, f->output_data.win32->top_pos);
+    }
+
+  /* If one of the icon positions was not set, preserve or default it.  */
+  if (EQ (icon_left, Qunbound) || ! INTEGERP (icon_left))
+    {
+      icon_left_no_change = 1;
+      icon_left = Fcdr (Fassq (Qicon_left, f->param_alist));
+      if (NILP (icon_left))
+	XSETINT (icon_left, 0);
+    }
+  if (EQ (icon_top, Qunbound) || ! INTEGERP (icon_top))
+    {
+      icon_top_no_change = 1;
+      icon_top = Fcdr (Fassq (Qicon_top, f->param_alist));
+      if (NILP (icon_top))
+	XSETINT (icon_top, 0);
+    }
+
+  /* Don't die if just one of these was set.  */
+  if (EQ (width, Qunbound))
+    XSETINT (width, FRAME_WIDTH (f));
+  if (EQ (height, Qunbound))
+    XSETINT (height, FRAME_HEIGHT (f));
+
+  /* Don't set these parameters unless they've been explicitly
+     specified.  The window might be mapped or resized while we're in
+     this function, and we don't want to override that unless the lisp
+     code has asked for it.
+
+     Don't set these parameters unless they actually differ from the
+     window's current parameters; the window may not actually exist
+     yet.  */
+  {
+    Lisp_Object frame;
+
+    check_frame_size (f, &height, &width);
+
+    XSETFRAME (frame, f);
+
+    if ((NUMBERP (width) && XINT (width) != FRAME_WIDTH (f))
+	|| (NUMBERP (height) && XINT (height) != FRAME_HEIGHT (f)))
+      Fset_frame_size (frame, width, height);
+
+    if ((!NILP (left) || !NILP (top))
+	&& ! (left_no_change && top_no_change)
+	&& ! (NUMBERP (left) && XINT (left) == f->output_data.win32->left_pos
+	      && NUMBERP (top) && XINT (top) == f->output_data.win32->top_pos))
+      {
+	int leftpos = 0;
+	int toppos = 0;
+
+	/* Record the signs.  */
+	f->output_data.win32->size_hint_flags &= ~ (XNegative | YNegative);
+	if (EQ (left, Qminus))
+	  f->output_data.win32->size_hint_flags |= XNegative;
+	else if (INTEGERP (left))
+	  {
+	    leftpos = XINT (left);
+	    if (leftpos < 0)
+	      f->output_data.win32->size_hint_flags |= XNegative;
+	  }
+	else if (CONSP (left) && EQ (XCONS (left)->car, Qminus)
+		 && CONSP (XCONS (left)->cdr)
+		 && INTEGERP (XCONS (XCONS (left)->cdr)->car))
+	  {
+	    leftpos = - XINT (XCONS (XCONS (left)->cdr)->car);
+	    f->output_data.win32->size_hint_flags |= XNegative;
+	  }
+	else if (CONSP (left) && EQ (XCONS (left)->car, Qplus)
+		 && CONSP (XCONS (left)->cdr)
+		 && INTEGERP (XCONS (XCONS (left)->cdr)->car))
+	  {
+	    leftpos = XINT (XCONS (XCONS (left)->cdr)->car);
+	  }
+
+	if (EQ (top, Qminus))
+	  f->output_data.win32->size_hint_flags |= YNegative;
+	else if (INTEGERP (top))
+	  {
+	    toppos = XINT (top);
+	    if (toppos < 0)
+	      f->output_data.win32->size_hint_flags |= YNegative;
+	  }
+	else if (CONSP (top) && EQ (XCONS (top)->car, Qminus)
+		 && CONSP (XCONS (top)->cdr)
+		 && INTEGERP (XCONS (XCONS (top)->cdr)->car))
+	  {
+	    toppos = - XINT (XCONS (XCONS (top)->cdr)->car);
+	    f->output_data.win32->size_hint_flags |= YNegative;
+	  }
+	else if (CONSP (top) && EQ (XCONS (top)->car, Qplus)
+		 && CONSP (XCONS (top)->cdr)
+		 && INTEGERP (XCONS (XCONS (top)->cdr)->car))
+	  {
+	    toppos = XINT (XCONS (XCONS (top)->cdr)->car);
+	  }
+
+
+	/* Store the numeric value of the position.  */
+	f->output_data.win32->top_pos = toppos;
+	f->output_data.win32->left_pos = leftpos;
+
+	f->output_data.win32->win_gravity = NorthWestGravity;
+
+	/* Actually set that position, and convert to absolute.  */
+	x_set_offset (f, leftpos, toppos, -1);
+      }
+
+    if ((!NILP (icon_left) || !NILP (icon_top))
+	&& ! (icon_left_no_change && icon_top_no_change))
+      x_wm_set_icon_position (f, XINT (icon_left), XINT (icon_top));
+  }
+}
+
+/* Store the screen positions of frame F into XPTR and YPTR.
+   These are the positions of the containing window manager window,
+   not Emacs's own window.  */
+
+void
+x_real_positions (f, xptr, yptr)
+     FRAME_PTR f;
+     int *xptr, *yptr;
+{
+  POINT pt;
+
+  {
+      RECT rect;
+      
+      GetClientRect(FRAME_WIN32_WINDOW(f), &rect);
+      AdjustWindowRect(&rect, f->output_data.win32->dwStyle, FRAME_EXTERNAL_MENU_BAR(f));
+      
+      pt.x = rect.left;
+      pt.y = rect.top;
+  }
+
+  ClientToScreen (FRAME_WIN32_WINDOW(f), &pt);
+
+  *xptr = pt.x;
+  *yptr = pt.y;
+}
+
+/* Insert a description of internally-recorded parameters of frame X
+   into the parameter alist *ALISTPTR that is to be given to the user.
+   Only parameters that are specific to Win32
+   and whose values are not correctly recorded in the frame's
+   param_alist need to be considered here.  */
+
+x_report_frame_params (f, alistptr)
+     struct frame *f;
+     Lisp_Object *alistptr;
+{
+  char buf[16];
+  Lisp_Object tem;
+
+  /* Represent negative positions (off the top or left screen edge)
+     in a way that Fmodify_frame_parameters will understand correctly.  */
+  XSETINT (tem, f->output_data.win32->left_pos);
+  if (f->output_data.win32->left_pos >= 0)
+    store_in_alist (alistptr, Qleft, tem);
+  else
+    store_in_alist (alistptr, Qleft, Fcons (Qplus, Fcons (tem, Qnil)));
+
+  XSETINT (tem, f->output_data.win32->top_pos);
+  if (f->output_data.win32->top_pos >= 0)
+    store_in_alist (alistptr, Qtop, tem);
+  else
+    store_in_alist (alistptr, Qtop, Fcons (Qplus, Fcons (tem, Qnil)));
+
+  store_in_alist (alistptr, Qborder_width,
+       	   make_number (f->output_data.win32->border_width));
+  store_in_alist (alistptr, Qinternal_border_width,
+       	   make_number (f->output_data.win32->internal_border_width));
+  sprintf (buf, "%ld", (long) FRAME_WIN32_WINDOW (f));
+  store_in_alist (alistptr, Qwindow_id,
+       	   build_string (buf));
+  store_in_alist (alistptr, Qicon_name, f->icon_name);
+  FRAME_SAMPLE_VISIBILITY (f);
+  store_in_alist (alistptr, Qvisibility,
+		  (FRAME_VISIBLE_P (f) ? Qt
+		   : FRAME_ICONIFIED_P (f) ? Qicon : Qnil));
+  store_in_alist (alistptr, Qdisplay,
+		  XCONS (FRAME_WIN32_DISPLAY_INFO (f)->name_list_element)->car);
+}
+
+
+#if 0
+DEFUN ("win32-rgb", Fwin32_rgb, Swin32_rgb, 3, 3, 0,
+       "Convert RGB numbers to a windows color reference.")
+    (red, green, blue)
+    Lisp_Object red, green, blue;
+{
+    Lisp_Object rgb;
+
+    CHECK_NUMBER (red, 0);
+    CHECK_NUMBER (green, 0);
+    CHECK_NUMBER (blue, 0);
+
+    XSET (rgb, Lisp_Int, RGB(XUINT(red), XUINT(green), XUINT(blue)));
+
+    return (rgb);
+}
+
+
+#else
+/* The default colors for the win32 color map */
+typedef struct colormap_t 
+{
+  char *name;
+  COLORREF colorref;
+} colormap_t;
+
+colormap_t win32_color_map[] = 
+{
+  {"snow"                      , RGB (255,250,250)},
+  {"ghost white"               , RGB (248,248,255)},
+  {"GhostWhite"                , RGB (248,248,255)},
+  {"white smoke"               , RGB (245,245,245)},
+  {"WhiteSmoke"                , RGB (245,245,245)},
+  {"gainsboro"                 , RGB (220,220,220)},
+  {"floral white"              , RGB (255,250,240)},
+  {"FloralWhite"               , RGB (255,250,240)},
+  {"old lace"                  , RGB (253,245,230)},
+  {"OldLace"                   , RGB (253,245,230)},
+  {"linen"                     , RGB (250,240,230)},
+  {"antique white"             , RGB (250,235,215)},
+  {"AntiqueWhite"              , RGB (250,235,215)},
+  {"papaya whip"               , RGB (255,239,213)},
+  {"PapayaWhip"                , RGB (255,239,213)},
+  {"blanched almond"           , RGB (255,235,205)},
+  {"BlanchedAlmond"            , RGB (255,235,205)},
+  {"bisque"                    , RGB (255,228,196)},
+  {"peach puff"                , RGB (255,218,185)},
+  {"PeachPuff"                 , RGB (255,218,185)},
+  {"navajo white"              , RGB (255,222,173)},
+  {"NavajoWhite"               , RGB (255,222,173)},
+  {"moccasin"                  , RGB (255,228,181)},
+  {"cornsilk"                  , RGB (255,248,220)},
+  {"ivory"                     , RGB (255,255,240)},
+  {"lemon chiffon"             , RGB (255,250,205)},
+  {"LemonChiffon"              , RGB (255,250,205)},
+  {"seashell"                  , RGB (255,245,238)},
+  {"honeydew"                  , RGB (240,255,240)},
+  {"mint cream"                , RGB (245,255,250)},
+  {"MintCream"                 , RGB (245,255,250)},
+  {"azure"                     , RGB (240,255,255)},
+  {"alice blue"                , RGB (240,248,255)},
+  {"AliceBlue"                 , RGB (240,248,255)},
+  {"lavender"                  , RGB (230,230,250)},
+  {"lavender blush"            , RGB (255,240,245)},
+  {"LavenderBlush"             , RGB (255,240,245)},
+  {"misty rose"                , RGB (255,228,225)},
+  {"MistyRose"                 , RGB (255,228,225)},
+  {"white"                     , RGB (255,255,255)},
+  {"black"                     , RGB (  0,  0,  0)},
+  {"dark slate gray"           , RGB ( 47, 79, 79)},
+  {"DarkSlateGray"             , RGB ( 47, 79, 79)},
+  {"dark slate grey"           , RGB ( 47, 79, 79)},
+  {"DarkSlateGrey"             , RGB ( 47, 79, 79)},
+  {"dim gray"                  , RGB (105,105,105)},
+  {"DimGray"                   , RGB (105,105,105)},
+  {"dim grey"                  , RGB (105,105,105)},
+  {"DimGrey"                   , RGB (105,105,105)},
+  {"slate gray"                , RGB (112,128,144)},
+  {"SlateGray"                 , RGB (112,128,144)},
+  {"slate grey"                , RGB (112,128,144)},
+  {"SlateGrey"                 , RGB (112,128,144)},
+  {"light slate gray"          , RGB (119,136,153)},
+  {"LightSlateGray"            , RGB (119,136,153)},
+  {"light slate grey"          , RGB (119,136,153)},
+  {"LightSlateGrey"            , RGB (119,136,153)},
+  {"gray"                      , RGB (190,190,190)},
+  {"grey"                      , RGB (190,190,190)},
+  {"light grey"                , RGB (211,211,211)},
+  {"LightGrey"                 , RGB (211,211,211)},
+  {"light gray"                , RGB (211,211,211)},
+  {"LightGray"                 , RGB (211,211,211)},
+  {"midnight blue"             , RGB ( 25, 25,112)},
+  {"MidnightBlue"              , RGB ( 25, 25,112)},
+  {"navy"                      , RGB (  0,  0,128)},
+  {"navy blue"                 , RGB (  0,  0,128)},
+  {"NavyBlue"                  , RGB (  0,  0,128)},
+  {"cornflower blue"           , RGB (100,149,237)},
+  {"CornflowerBlue"            , RGB (100,149,237)},
+  {"dark slate blue"           , RGB ( 72, 61,139)},
+  {"DarkSlateBlue"             , RGB ( 72, 61,139)},
+  {"slate blue"                , RGB (106, 90,205)},
+  {"SlateBlue"                 , RGB (106, 90,205)},
+  {"medium slate blue"         , RGB (123,104,238)},
+  {"MediumSlateBlue"           , RGB (123,104,238)},
+  {"light slate blue"          , RGB (132,112,255)},
+  {"LightSlateBlue"            , RGB (132,112,255)},
+  {"medium blue"               , RGB (  0,  0,205)},
+  {"MediumBlue"                , RGB (  0,  0,205)},
+  {"royal blue"                , RGB ( 65,105,225)},
+  {"RoyalBlue"                 , RGB ( 65,105,225)},
+  {"blue"                      , RGB (  0,  0,255)},
+  {"dodger blue"               , RGB ( 30,144,255)},
+  {"DodgerBlue"                , RGB ( 30,144,255)},
+  {"deep sky blue"             , RGB (  0,191,255)},
+  {"DeepSkyBlue"               , RGB (  0,191,255)},
+  {"sky blue"                  , RGB (135,206,235)},
+  {"SkyBlue"                   , RGB (135,206,235)},
+  {"light sky blue"            , RGB (135,206,250)},
+  {"LightSkyBlue"              , RGB (135,206,250)},
+  {"steel blue"                , RGB ( 70,130,180)},
+  {"SteelBlue"                 , RGB ( 70,130,180)},
+  {"light steel blue"          , RGB (176,196,222)},
+  {"LightSteelBlue"            , RGB (176,196,222)},
+  {"light blue"                , RGB (173,216,230)},
+  {"LightBlue"                 , RGB (173,216,230)},
+  {"powder blue"               , RGB (176,224,230)},
+  {"PowderBlue"                , RGB (176,224,230)},
+  {"pale turquoise"            , RGB (175,238,238)},
+  {"PaleTurquoise"             , RGB (175,238,238)},
+  {"dark turquoise"            , RGB (  0,206,209)},
+  {"DarkTurquoise"             , RGB (  0,206,209)},
+  {"medium turquoise"          , RGB ( 72,209,204)},
+  {"MediumTurquoise"           , RGB ( 72,209,204)},
+  {"turquoise"                 , RGB ( 64,224,208)},
+  {"cyan"                      , RGB (  0,255,255)},
+  {"light cyan"                , RGB (224,255,255)},
+  {"LightCyan"                 , RGB (224,255,255)},
+  {"cadet blue"                , RGB ( 95,158,160)},
+  {"CadetBlue"                 , RGB ( 95,158,160)},
+  {"medium aquamarine"         , RGB (102,205,170)},
+  {"MediumAquamarine"          , RGB (102,205,170)},
+  {"aquamarine"                , RGB (127,255,212)},
+  {"dark green"                , RGB (  0,100,  0)},
+  {"DarkGreen"                 , RGB (  0,100,  0)},
+  {"dark olive green"          , RGB ( 85,107, 47)},
+  {"DarkOliveGreen"            , RGB ( 85,107, 47)},
+  {"dark sea green"            , RGB (143,188,143)},
+  {"DarkSeaGreen"              , RGB (143,188,143)},
+  {"sea green"                 , RGB ( 46,139, 87)},
+  {"SeaGreen"                  , RGB ( 46,139, 87)},
+  {"medium sea green"          , RGB ( 60,179,113)},
+  {"MediumSeaGreen"            , RGB ( 60,179,113)},
+  {"light sea green"           , RGB ( 32,178,170)},
+  {"LightSeaGreen"             , RGB ( 32,178,170)},
+  {"pale green"                , RGB (152,251,152)},
+  {"PaleGreen"                 , RGB (152,251,152)},
+  {"spring green"              , RGB (  0,255,127)},
+  {"SpringGreen"               , RGB (  0,255,127)},
+  {"lawn green"                , RGB (124,252,  0)},
+  {"LawnGreen"                 , RGB (124,252,  0)},
+  {"green"                     , RGB (  0,255,  0)},
+  {"chartreuse"                , RGB (127,255,  0)},
+  {"medium spring green"       , RGB (  0,250,154)},
+  {"MediumSpringGreen"         , RGB (  0,250,154)},
+  {"green yellow"              , RGB (173,255, 47)},
+  {"GreenYellow"               , RGB (173,255, 47)},
+  {"lime green"                , RGB ( 50,205, 50)},
+  {"LimeGreen"                 , RGB ( 50,205, 50)},
+  {"yellow green"              , RGB (154,205, 50)},
+  {"YellowGreen"               , RGB (154,205, 50)},
+  {"forest green"              , RGB ( 34,139, 34)},
+  {"ForestGreen"               , RGB ( 34,139, 34)},
+  {"olive drab"                , RGB (107,142, 35)},
+  {"OliveDrab"                 , RGB (107,142, 35)},
+  {"dark khaki"                , RGB (189,183,107)},
+  {"DarkKhaki"                 , RGB (189,183,107)},
+  {"khaki"                     , RGB (240,230,140)},
+  {"pale goldenrod"            , RGB (238,232,170)},
+  {"PaleGoldenrod"             , RGB (238,232,170)},
+  {"light goldenrod yellow"    , RGB (250,250,210)},
+  {"LightGoldenrodYellow"      , RGB (250,250,210)},
+  {"light yellow"              , RGB (255,255,224)},
+  {"LightYellow"               , RGB (255,255,224)},
+  {"yellow"                    , RGB (255,255,  0)},
+  {"gold"                      , RGB (255,215,  0)},
+  {"light goldenrod"           , RGB (238,221,130)},
+  {"LightGoldenrod"            , RGB (238,221,130)},
+  {"goldenrod"                 , RGB (218,165, 32)},
+  {"dark goldenrod"            , RGB (184,134, 11)},
+  {"DarkGoldenrod"             , RGB (184,134, 11)},
+  {"rosy brown"                , RGB (188,143,143)},
+  {"RosyBrown"                 , RGB (188,143,143)},
+  {"indian red"                , RGB (205, 92, 92)},
+  {"IndianRed"                 , RGB (205, 92, 92)},
+  {"saddle brown"              , RGB (139, 69, 19)},
+  {"SaddleBrown"               , RGB (139, 69, 19)},
+  {"sienna"                    , RGB (160, 82, 45)},
+  {"peru"                      , RGB (205,133, 63)},
+  {"burlywood"                 , RGB (222,184,135)},
+  {"beige"                     , RGB (245,245,220)},
+  {"wheat"                     , RGB (245,222,179)},
+  {"sandy brown"               , RGB (244,164, 96)},
+  {"SandyBrown"                , RGB (244,164, 96)},
+  {"tan"                       , RGB (210,180,140)},
+  {"chocolate"                 , RGB (210,105, 30)},
+  {"firebrick"                 , RGB (178,34, 34)},
+  {"brown"                     , RGB (165,42, 42)},
+  {"dark salmon"               , RGB (233,150,122)},
+  {"DarkSalmon"                , RGB (233,150,122)},
+  {"salmon"                    , RGB (250,128,114)},
+  {"light salmon"              , RGB (255,160,122)},
+  {"LightSalmon"               , RGB (255,160,122)},
+  {"orange"                    , RGB (255,165,  0)},
+  {"dark orange"               , RGB (255,140,  0)},
+  {"DarkOrange"                , RGB (255,140,  0)},
+  {"coral"                     , RGB (255,127, 80)},
+  {"light coral"               , RGB (240,128,128)},
+  {"LightCoral"                , RGB (240,128,128)},
+  {"tomato"                    , RGB (255, 99, 71)},
+  {"orange red"                , RGB (255, 69,  0)},
+  {"OrangeRed"                 , RGB (255, 69,  0)},
+  {"red"                       , RGB (255,  0,  0)},
+  {"hot pink"                  , RGB (255,105,180)},
+  {"HotPink"                   , RGB (255,105,180)},
+  {"deep pink"                 , RGB (255, 20,147)},
+  {"DeepPink"                  , RGB (255, 20,147)},
+  {"pink"                      , RGB (255,192,203)},
+  {"light pink"                , RGB (255,182,193)},
+  {"LightPink"                 , RGB (255,182,193)},
+  {"pale violet red"           , RGB (219,112,147)},
+  {"PaleVioletRed"             , RGB (219,112,147)},
+  {"maroon"                    , RGB (176, 48, 96)},
+  {"medium violet red"         , RGB (199, 21,133)},
+  {"MediumVioletRed"           , RGB (199, 21,133)},
+  {"violet red"                , RGB (208, 32,144)},
+  {"VioletRed"                 , RGB (208, 32,144)},
+  {"magenta"                   , RGB (255,  0,255)},
+  {"violet"                    , RGB (238,130,238)},
+  {"plum"                      , RGB (221,160,221)},
+  {"orchid"                    , RGB (218,112,214)},
+  {"medium orchid"             , RGB (186, 85,211)},
+  {"MediumOrchid"              , RGB (186, 85,211)},
+  {"dark orchid"               , RGB (153, 50,204)},
+  {"DarkOrchid"                , RGB (153, 50,204)},
+  {"dark violet"               , RGB (148,  0,211)},
+  {"DarkViolet"                , RGB (148,  0,211)},
+  {"blue violet"               , RGB (138, 43,226)},
+  {"BlueViolet"                , RGB (138, 43,226)},
+  {"purple"                    , RGB (160, 32,240)},
+  {"medium purple"             , RGB (147,112,219)},
+  {"MediumPurple"              , RGB (147,112,219)},
+  {"thistle"                   , RGB (216,191,216)},
+  {"gray0"                     , RGB (  0,  0,  0)},
+  {"grey0"                     , RGB (  0,  0,  0)},
+  {"dark grey"                 , RGB (169,169,169)},
+  {"DarkGrey"                  , RGB (169,169,169)},
+  {"dark gray"                 , RGB (169,169,169)},
+  {"DarkGray"                  , RGB (169,169,169)},
+  {"dark blue"                 , RGB (  0,  0,139)},
+  {"DarkBlue"                  , RGB (  0,  0,139)},
+  {"dark cyan"                 , RGB (  0,139,139)},
+  {"DarkCyan"                  , RGB (  0,139,139)},
+  {"dark magenta"              , RGB (139,  0,139)},
+  {"DarkMagenta"               , RGB (139,  0,139)},
+  {"dark red"                  , RGB (139,  0,  0)},
+  {"DarkRed"                   , RGB (139,  0,  0)},
+  {"light green"               , RGB (144,238,144)},
+  {"LightGreen"                , RGB (144,238,144)},
+};
+
+DEFUN ("win32-default-color-map", Fwin32_default_color_map, Swin32_default_color_map,
+       0, 0, 0, "Return the default color map.")
+     ()
+{
+  int i;
+  colormap_t *pc = win32_color_map;
+  Lisp_Object cmap;
+  
+  BLOCK_INPUT;
+  
+  cmap = Qnil;
+  
+  for (i = 0; i < sizeof (win32_color_map) / sizeof (win32_color_map[0]); 
+       pc++, i++)
+    cmap = Fcons (Fcons (build_string (pc->name),
+			 make_number (pc->colorref)),
+		  cmap);
+  
+  UNBLOCK_INPUT;
+  
+  return (cmap);
+}
+#endif
+
+Lisp_Object 
+win32_to_x_color (rgb)
+     Lisp_Object rgb;
+{
+  Lisp_Object color;
+  
+  CHECK_NUMBER (rgb, 0);
+  
+  BLOCK_INPUT;
+  
+  color = Frassq (rgb, Vwin32_color_map);
+  
+  UNBLOCK_INPUT;
+  
+  if (!NILP (color))
+    return (Fcar (color));
+  else
+    return Qnil;
+}
+
+COLORREF 
+x_to_win32_color (colorname)
+     char * colorname;
+{
+  register Lisp_Object tail, ret = Qnil;
+  
+  BLOCK_INPUT;
+  
+  for (tail = Vwin32_color_map; !NILP (tail); tail = Fcdr (tail))
+    {
+      register Lisp_Object elt, tem;
+
+      elt = Fcar (tail);
+      if (!CONSP (elt)) continue;
+
+      tem = Fcar (elt);
+
+      if (lstrcmpi (XSTRING (tem)->data, colorname) == 0)
+	{
+	  ret = XUINT(Fcdr (elt));
+	  break;
+	}
+
+      QUIT;
+    }
+  
+  UNBLOCK_INPUT;
+  
+  return ret;
+}
+
+/* Decide if color named COLOR is valid for the display associated with
+   the selected frame; if so, return the rgb values in COLOR_DEF.
+   If ALLOC is nonzero, allocate a new colormap cell.  */
+
+int
+defined_color (f, color, color_def, alloc)
+     FRAME_PTR f;
+     char *color;
+     COLORREF *color_def;
+     int alloc;
+{
+  register Lisp_Object tem;
+
+  tem = x_to_win32_color (color);
+
+  if (!NILP (tem)) 
+    {
+      *color_def = XUINT (tem);
+      return 1;
+    } 
+  else 
+    {
+      return 0;
+    }
+}
+
+/* Given a string ARG naming a color, compute a pixel value from it
+   suitable for screen F.
+   If F is not a color screen, return DEF (default) regardless of what
+   ARG says.  */
+
+int
+x_decode_color (f, arg, def)
+     FRAME_PTR f;
+     Lisp_Object arg;
+     int def;
+{
+  COLORREF cdef;
+
+  CHECK_STRING (arg, 0);
+
+  if (strcmp (XSTRING (arg)->data, "black") == 0)
+    return BLACK_PIX_DEFAULT (f);
+  else if (strcmp (XSTRING (arg)->data, "white") == 0)
+    return WHITE_PIX_DEFAULT (f);
+
+  if ((FRAME_WIN32_DISPLAY_INFO (f)->n_planes * FRAME_WIN32_DISPLAY_INFO (f)->n_cbits) == 1)
+    return def;
+
+  /* defined_color is responsible for coping with failures
+     by looking for a near-miss.  */
+  if (defined_color (f, XSTRING (arg)->data, &cdef, 1))
+    return cdef;
+
+  /* defined_color failed; return an ultimate default.  */
+  return def;
+}
+
+/* Functions called only from `x_set_frame_param'
+   to set individual parameters.
+
+   If FRAME_WIN32_WINDOW (f) is 0,
+   the frame is being created and its window does not exist yet.
+   In that case, just record the parameter's new value
+   in the standard place; do not attempt to change the window.  */
+
+void
+x_set_foreground_color (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  f->output_data.win32->foreground_pixel
+    = x_decode_color (f, arg, BLACK_PIX_DEFAULT (f));
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    {
+      recompute_basic_faces (f);
+      if (FRAME_VISIBLE_P (f))
+        redraw_frame (f);
+    }
+}
+
+void
+x_set_background_color (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  Pixmap temp;
+  int mask;
+
+  f->output_data.win32->background_pixel
+    = x_decode_color (f, arg, WHITE_PIX_DEFAULT (f));
+
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    {
+      SetWindowLong (FRAME_WIN32_WINDOW (f), WND_BACKGROUND_INDEX, f->output_data.win32->background_pixel);
+
+      recompute_basic_faces (f);
+
+      if (FRAME_VISIBLE_P (f))
+        redraw_frame (f);
+    }
+}
+
+void
+x_set_mouse_color (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+#if 0
+  Cursor cursor, nontext_cursor, mode_cursor, cross_cursor;
+#endif
+  int mask_color;
+
+  if (!EQ (Qnil, arg))
+    f->output_data.win32->mouse_pixel
+      = x_decode_color (f, arg, BLACK_PIX_DEFAULT (f));
+  mask_color = f->output_data.win32->background_pixel;
+				/* No invisible pointers.  */
+  if (mask_color == f->output_data.win32->mouse_pixel
+	&& mask_color == f->output_data.win32->background_pixel)
+    f->output_data.win32->mouse_pixel = f->output_data.win32->foreground_pixel;
+
+#if 0
+  BLOCK_INPUT;
+
+  /* It's not okay to crash if the user selects a screwy cursor.  */
+  x_catch_errors (FRAME_WIN32_DISPLAY (f));
+
+  if (!EQ (Qnil, Vx_pointer_shape))
+    {
+      CHECK_NUMBER (Vx_pointer_shape, 0);
+      cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f), XINT (Vx_pointer_shape));
+    }
+  else
+    cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f), XC_xterm);
+  x_check_errors (FRAME_WIN32_DISPLAY (f), "bad text pointer cursor: %s");
+
+  if (!EQ (Qnil, Vx_nontext_pointer_shape))
+    {
+      CHECK_NUMBER (Vx_nontext_pointer_shape, 0);
+      nontext_cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f),
+					  XINT (Vx_nontext_pointer_shape));
+    }
+  else
+    nontext_cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f), XC_left_ptr);
+  x_check_errors (FRAME_WIN32_DISPLAY (f), "bad nontext pointer cursor: %s");
+
+  if (!EQ (Qnil, Vx_mode_pointer_shape))
+    {
+      CHECK_NUMBER (Vx_mode_pointer_shape, 0);
+      mode_cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f),
+				       XINT (Vx_mode_pointer_shape));
+    }
+  else
+    mode_cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f), XC_xterm);
+  x_check_errors (FRAME_WIN32_DISPLAY (f), "bad modeline pointer cursor: %s");
+
+  if (!EQ (Qnil, Vx_sensitive_text_pointer_shape))
+    {
+      CHECK_NUMBER (Vx_sensitive_text_pointer_shape, 0);
+      cross_cursor
+	= XCreateFontCursor (FRAME_WIN32_DISPLAY (f),
+			     XINT (Vx_sensitive_text_pointer_shape));
+    }
+  else
+    cross_cursor = XCreateFontCursor (FRAME_WIN32_DISPLAY (f), XC_crosshair);
+
+  /* Check and report errors with the above calls.  */
+  x_check_errors (FRAME_WIN32_DISPLAY (f), "can't set cursor shape: %s");
+  x_uncatch_errors (FRAME_WIN32_DISPLAY (f));
+
+  {
+    XColor fore_color, back_color;
+
+    fore_color.pixel = f->output_data.win32->mouse_pixel;
+    back_color.pixel = mask_color;
+    XQueryColor (FRAME_WIN32_DISPLAY (f),
+		 DefaultColormap (FRAME_WIN32_DISPLAY (f),
+				  DefaultScreen (FRAME_WIN32_DISPLAY (f))),
+		 &fore_color);
+    XQueryColor (FRAME_WIN32_DISPLAY (f),
+		 DefaultColormap (FRAME_WIN32_DISPLAY (f),
+				  DefaultScreen (FRAME_WIN32_DISPLAY (f))),
+		 &back_color);
+    XRecolorCursor (FRAME_WIN32_DISPLAY (f), cursor,
+		    &fore_color, &back_color);
+    XRecolorCursor (FRAME_WIN32_DISPLAY (f), nontext_cursor,
+		    &fore_color, &back_color);
+    XRecolorCursor (FRAME_WIN32_DISPLAY (f), mode_cursor,
+		    &fore_color, &back_color);
+    XRecolorCursor (FRAME_WIN32_DISPLAY (f), cross_cursor,
+                    &fore_color, &back_color);
+  }
+
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    {
+      XDefineCursor (FRAME_WIN32_DISPLAY (f), FRAME_WIN32_WINDOW (f), cursor);
+    }
+
+  if (cursor != f->output_data.win32->text_cursor && f->output_data.win32->text_cursor != 0)
+    XFreeCursor (FRAME_WIN32_DISPLAY (f), f->output_data.win32->text_cursor);
+  f->output_data.win32->text_cursor = cursor;
+
+  if (nontext_cursor != f->output_data.win32->nontext_cursor
+      && f->output_data.win32->nontext_cursor != 0)
+    XFreeCursor (FRAME_WIN32_DISPLAY (f), f->output_data.win32->nontext_cursor);
+  f->output_data.win32->nontext_cursor = nontext_cursor;
+
+  if (mode_cursor != f->output_data.win32->modeline_cursor
+      && f->output_data.win32->modeline_cursor != 0)
+    XFreeCursor (FRAME_WIN32_DISPLAY (f), f->output_data.win32->modeline_cursor);
+  f->output_data.win32->modeline_cursor = mode_cursor;
+  if (cross_cursor != f->output_data.win32->cross_cursor
+      && f->output_data.win32->cross_cursor != 0)
+    XFreeCursor (FRAME_WIN32_DISPLAY (f), f->output_data.win32->cross_cursor);
+  f->output_data.win32->cross_cursor = cross_cursor;
+
+  XFlush (FRAME_WIN32_DISPLAY (f));
+  UNBLOCK_INPUT;
+#endif
+}
+
+void
+x_set_cursor_color (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  unsigned long fore_pixel;
+
+  if (!EQ (Vx_cursor_fore_pixel, Qnil))
+    fore_pixel = x_decode_color (f, Vx_cursor_fore_pixel,
+				 WHITE_PIX_DEFAULT (f));
+  else
+    fore_pixel = f->output_data.win32->background_pixel;
+  f->output_data.win32->cursor_pixel = x_decode_color (f, arg, BLACK_PIX_DEFAULT (f));
+  
+  /* Make sure that the cursor color differs from the background color.  */
+  if (f->output_data.win32->cursor_pixel == f->output_data.win32->background_pixel)
+    {
+      f->output_data.win32->cursor_pixel = f->output_data.win32->mouse_pixel;
+      if (f->output_data.win32->cursor_pixel == fore_pixel)
+	fore_pixel = f->output_data.win32->background_pixel;
+    }
+  f->output_data.win32->cursor_foreground_pixel = fore_pixel;
+
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    {
+      if (FRAME_VISIBLE_P (f))
+	{
+	  x_display_cursor (f, 0);
+	  x_display_cursor (f, 1);
+	}
+    }
+}
+
+/* Set the border-color of frame F to value described by ARG.
+   ARG can be a string naming a color.
+   The border-color is used for the border that is drawn by the server.
+   Note that this does not fully take effect if done before
+   F has a window; it must be redone when the window is created.  */
+
+void
+x_set_border_color (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  unsigned char *str;
+  int pix;
+
+  CHECK_STRING (arg, 0);
+  str = XSTRING (arg)->data;
+
+  pix = x_decode_color (f, arg, BLACK_PIX_DEFAULT (f));
+
+  x_set_border_pixel (f, pix);
+}
+
+/* Set the border-color of frame F to pixel value PIX.
+   Note that this does not fully take effect if done before
+   F has an window.  */
+
+x_set_border_pixel (f, pix)
+     struct frame *f;
+     int pix;
+{
+  f->output_data.win32->border_pixel = pix;
+
+  if (FRAME_WIN32_WINDOW (f) != 0 && f->output_data.win32->border_width > 0)
+    {
+      if (FRAME_VISIBLE_P (f))
+        redraw_frame (f);
+    }
+}
+
+void
+x_set_cursor_type (f, arg, oldval)
+     FRAME_PTR f;
+     Lisp_Object arg, oldval;
+{
+  if (EQ (arg, Qbar))
+    {
+      FRAME_DESIRED_CURSOR (f) = bar_cursor;
+      f->output_data.win32->cursor_width = 2;
+    }
+  else if (CONSP (arg) && EQ (XCONS (arg)->car, Qbar)
+	   && INTEGERP (XCONS (arg)->cdr))
+    {
+      FRAME_DESIRED_CURSOR (f) = bar_cursor;
+      f->output_data.win32->cursor_width = XINT (XCONS (arg)->cdr);
+    }
+  else
+    /* Treat anything unknown as "box cursor".
+       It was bad to signal an error; people have trouble fixing
+       .Xdefaults with Emacs, when it has something bad in it.  */
+    FRAME_DESIRED_CURSOR (f) = filled_box_cursor;
+
+  /* Make sure the cursor gets redrawn.  This is overkill, but how
+     often do people change cursor types?  */
+  update_mode_lines++;
+}
+
+void
+x_set_icon_type (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+#if 0
+  Lisp_Object tem;
+  int result;
+
+  if (STRINGP (arg))
+    {
+      if (STRINGP (oldval) && EQ (Fstring_equal (oldval, arg), Qt))
+	return;
+    }
+  else if (!STRINGP (oldval) && EQ (oldval, Qnil) == EQ (arg, Qnil))
+    return;
+
+  BLOCK_INPUT;
+  if (NILP (arg))
+    result = x_text_icon (f,
+			  (char *) XSTRING ((!NILP (f->icon_name)
+					     ? f->icon_name
+					     : f->name))->data);
+  else
+    result = x_bitmap_icon (f, arg);
+
+  if (result)
+    {
+      UNBLOCK_INPUT;
+      error ("No icon window available");
+    }
+
+  /* If the window was unmapped (and its icon was mapped),
+     the new icon is not mapped, so map the window in its stead.  */
+  if (FRAME_VISIBLE_P (f))
+    {
+#ifdef USE_X_TOOLKIT
+      XtPopup (f->output_data.win32->widget, XtGrabNone);
+#endif
+      XMapWindow (FRAME_WIN32_DISPLAY (f), FRAME_WIN32_WINDOW (f));
+    }
+
+  XFlush (FRAME_WIN32_DISPLAY (f));
+  UNBLOCK_INPUT;
+#endif
+}
+
+/* Return non-nil if frame F wants a bitmap icon.  */
+
+Lisp_Object
+x_icon_type (f)
+     FRAME_PTR f;
+{
+  Lisp_Object tem;
+
+  tem = assq_no_quit (Qicon_type, f->param_alist);
+  if (CONSP (tem))
+    return XCONS (tem)->cdr;
+  else
+    return Qnil;
+}
+
+void
+x_set_icon_name (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  Lisp_Object tem;
+  int result;
+
+  if (STRINGP (arg))
+    {
+      if (STRINGP (oldval) && EQ (Fstring_equal (oldval, arg), Qt))
+	return;
+    }
+  else if (!STRINGP (oldval) && EQ (oldval, Qnil) == EQ (arg, Qnil))
+    return;
+
+  f->icon_name = arg;
+
+#if 0
+  if (f->output_data.win32->icon_bitmap != 0)
+    return;
+
+  BLOCK_INPUT;
+
+  result = x_text_icon (f,
+			 (char *) XSTRING ((!NILP (f->icon_name)
+					   ? f->icon_name
+					   : f->name))->data);
+
+  if (result)
+    {
+      UNBLOCK_INPUT;
+      error ("No icon window available");
+    }
+
+  /* If the window was unmapped (and its icon was mapped),
+     the new icon is not mapped, so map the window in its stead.  */
+  if (FRAME_VISIBLE_P (f))
+    {
+#ifdef USE_X_TOOLKIT
+      XtPopup (f->output_data.win32->widget, XtGrabNone);
+#endif
+      XMapWindow (FRAME_WIN32_DISPLAY (f), FRAME_WIN32_WINDOW (f));
+    }
+
+  XFlush (FRAME_WIN32_DISPLAY (f));
+  UNBLOCK_INPUT;
+#endif
+}
+
+extern Lisp_Object x_new_font ();
+
+void
+x_set_font (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  Lisp_Object result;
+
+  CHECK_STRING (arg, 1);
+
+  BLOCK_INPUT;
+  result = x_new_font (f, XSTRING (arg)->data);
+  UNBLOCK_INPUT;
+  
+  if (EQ (result, Qnil))
+    error ("Font \"%s\" is not defined", XSTRING (arg)->data);
+  else if (EQ (result, Qt))
+    error ("the characters of the given font have varying widths");
+  else if (STRINGP (result))
+    {
+      recompute_basic_faces (f);
+      store_frame_param (f, Qfont, result);
+    }
+  else
+    abort ();
+}
+
+void
+x_set_border_width (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  CHECK_NUMBER (arg, 0);
+
+  if (XINT (arg) == f->output_data.win32->border_width)
+    return;
+
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    error ("Cannot change the border width of a window");
+
+  f->output_data.win32->border_width = XINT (arg);
+}
+
+void
+x_set_internal_border_width (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  int mask;
+  int old = f->output_data.win32->internal_border_width;
+
+  CHECK_NUMBER (arg, 0);
+  f->output_data.win32->internal_border_width = XINT (arg);
+  if (f->output_data.win32->internal_border_width < 0)
+    f->output_data.win32->internal_border_width = 0;
+
+  if (f->output_data.win32->internal_border_width == old)
+    return;
+
+  if (FRAME_WIN32_WINDOW (f) != 0)
+    {
+      BLOCK_INPUT;
+      x_set_window_size (f, 0, f->width, f->height);
+      UNBLOCK_INPUT;
+      SET_FRAME_GARBAGED (f);
+    }
+}
+
+void
+x_set_visibility (f, value, oldval)
+     struct frame *f;
+     Lisp_Object value, oldval;
+{
+  Lisp_Object frame;
+  XSETFRAME (frame, f);
+
+  if (NILP (value))
+    Fmake_frame_invisible (frame, Qt);
+  else if (EQ (value, Qicon))
+    Ficonify_frame (frame);
+  else
+    Fmake_frame_visible (frame);
+}
+
+void
+x_set_menu_bar_lines (f, value, oldval)
+     struct frame *f;
+     Lisp_Object value, oldval;
+{
+  int nlines;
+  int olines = FRAME_MENU_BAR_LINES (f);
+
+  /* Right now, menu bars don't work properly in minibuf-only frames;
+     most of the commands try to apply themselves to the minibuffer
+     frame itslef, and get an error because you can't switch buffers
+     in or split the minibuffer window.  */
+  if (FRAME_MINIBUF_ONLY_P (f))
+    return;
+
+  if (INTEGERP (value))
+    nlines = XINT (value);
+  else
+    nlines = 0;
+
+  FRAME_MENU_BAR_LINES (f) = 0;
+  if (nlines)
+    FRAME_EXTERNAL_MENU_BAR (f) = 1;
+  else
+    {
+      if (FRAME_EXTERNAL_MENU_BAR (f) == 1)
+	free_frame_menubar (f);
+      FRAME_EXTERNAL_MENU_BAR (f) = 0;
+    }
+}
+
+/* Change the name of frame F to NAME.  If NAME is nil, set F's name to
+       win32_id_name.
+
+   If EXPLICIT is non-zero, that indicates that lisp code is setting the
+       name; if NAME is a string, set F's name to NAME and set
+       F->explicit_name; if NAME is Qnil, then clear F->explicit_name.
+
+   If EXPLICIT is zero, that indicates that Emacs redisplay code is
+       suggesting a new name, which lisp code should override; if
+       F->explicit_name is set, ignore the new name; otherwise, set it.  */
+
+void
+x_set_name (f, name, explicit)
+     struct frame *f;
+     Lisp_Object name;
+     int explicit;
+{
+  /* Make sure that requests from lisp code override requests from 
+     Emacs redisplay code.  */
+  if (explicit)
+    {
+      /* If we're switching from explicit to implicit, we had better
+	 update the mode lines and thereby update the title.  */
+      if (f->explicit_name && NILP (name))
+	update_mode_lines = 1;
+
+      f->explicit_name = ! NILP (name);
+    }
+  else if (f->explicit_name)
+    return;
+
+  /* If NAME is nil, set the name to the win32_id_name.  */
+  if (NILP (name))
+    {
+      /* Check for no change needed in this very common case
+	 before we do any consing.  */
+      if (!strcmp (FRAME_WIN32_DISPLAY_INFO (f)->win32_id_name,
+		   XSTRING (f->name)->data))
+	return;
+      name = build_string (FRAME_WIN32_DISPLAY_INFO (f)->win32_id_name);
+    }
+  else
+    CHECK_STRING (name, 0);
+
+  /* Don't change the name if it's already NAME.  */
+  if (! NILP (Fstring_equal (name, f->name)))
+    return;
+
+  if (FRAME_WIN32_WINDOW (f))
+    {
+      BLOCK_INPUT;
+      SetWindowText(FRAME_WIN32_WINDOW (f), XSTRING (name)->data);
+      UNBLOCK_INPUT;
+    }
+
+  f->name = name;
+}
+
+/* This function should be called when the user's lisp code has
+   specified a name for the frame; the name will override any set by the
+   redisplay code.  */
+void
+x_explicitly_set_name (f, arg, oldval)
+     FRAME_PTR f;
+     Lisp_Object arg, oldval;
+{
+  x_set_name (f, arg, 1);
+}
+
+/* This function should be called by Emacs redisplay code to set the
+   name; names set this way will never override names set by the user's
+   lisp code.  */
+void
+x_implicitly_set_name (f, arg, oldval)
+     FRAME_PTR f;
+     Lisp_Object arg, oldval;
+{
+  x_set_name (f, arg, 0);
+}
+
+void
+x_set_autoraise (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  f->auto_raise = !EQ (Qnil, arg);
+}
+
+void
+x_set_autolower (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  f->auto_lower = !EQ (Qnil, arg);
+}
+
+void
+x_set_unsplittable (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  f->no_split = !NILP (arg);
+}
+
+void
+x_set_vertical_scroll_bars (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  if (NILP (arg) != ! FRAME_HAS_VERTICAL_SCROLL_BARS (f))
+    {
+      FRAME_HAS_VERTICAL_SCROLL_BARS (f) = ! NILP (arg);
+
+      /* We set this parameter before creating the window for the
+	 frame, so we can get the geometry right from the start.
+	 However, if the window hasn't been created yet, we shouldn't
+	 call x_set_window_size.  */
+      if (FRAME_WIN32_WINDOW (f))
+	x_set_window_size (f, 0, FRAME_WIDTH (f), FRAME_HEIGHT (f));
+    }
+}
+
+void
+x_set_scroll_bar_width (f, arg, oldval)
+     struct frame *f;
+     Lisp_Object arg, oldval;
+{
+  if (NILP (arg))
+    {
+      FRAME_SCROLL_BAR_PIXEL_WIDTH (f) = 0;
+      FRAME_SCROLL_BAR_COLS (f) = 2;
+    }
+  else if (INTEGERP (arg) && XINT (arg) > 0
+	   && XFASTINT (arg) != FRAME_SCROLL_BAR_PIXEL_WIDTH (f))
+    {
+      int wid = FONT_WIDTH (f->output_data.win32->font);
+      FRAME_SCROLL_BAR_PIXEL_WIDTH (f) = XFASTINT (arg);
+      FRAME_SCROLL_BAR_COLS (f) = (XFASTINT (arg) + wid-1) / wid;
+      if (FRAME_WIN32_WINDOW (f))
+	x_set_window_size (f, 0, FRAME_WIDTH (f), FRAME_HEIGHT (f));
+    }
+}
+
+/* Subroutines of creating an frame.  */
+
+/* Make sure that Vx_resource_name is set to a reasonable value.
+   Fix it up, or set it to `emacs' if it is too hopeless.  */
+
+static void
+validate_x_resource_name ()
+{
+  int len;
+  /* Number of valid characters in the resource name.  */
+  int good_count = 0;
+  /* Number of invalid characters in the resource name.  */
+  int bad_count = 0;
+  Lisp_Object new;
+  int i;
+
+  if (STRINGP (Vx_resource_name))
+    {
+      unsigned char *p = XSTRING (Vx_resource_name)->data;
+      int i;
+
+      len = XSTRING (Vx_resource_name)->size;
+
+      /* Only letters, digits, - and _ are valid in resource names.
+	 Count the valid characters and count the invalid ones.  */
+      for (i = 0; i < len; i++)
+	{
+	  int c = p[i];
+	  if (! ((c >= 'a' && c <= 'z')
+		 || (c >= 'A' && c <= 'Z')
+		 || (c >= '0' && c <= '9')
+		 || c == '-' || c == '_'))
+	    bad_count++;
+	  else
+	    good_count++;
+	}
+    }
+  else
+    /* Not a string => completely invalid.  */
+    bad_count = 5, good_count = 0;
+
+  /* If name is valid already, return.  */
+  if (bad_count == 0)
+    return;
+
+  /* If name is entirely invalid, or nearly so, use `emacs'.  */
+  if (good_count == 0
+      || (good_count == 1 && bad_count > 0))
+    {
+      Vx_resource_name = build_string ("emacs");
+      return;
+    }
+
+  /* Name is partly valid.  Copy it and replace the invalid characters
+     with underscores.  */
+
+  Vx_resource_name = new = Fcopy_sequence (Vx_resource_name);
+
+  for (i = 0; i < len; i++)
+    {
+      int c = XSTRING (new)->data[i];
+      if (! ((c >= 'a' && c <= 'z')
+	     || (c >= 'A' && c <= 'Z')
+	     || (c >= '0' && c <= '9')
+	     || c == '-' || c == '_'))
+	XSTRING (new)->data[i] = '_';
+    }
+}
+
+
+extern char *x_get_string_resource ();
+
+DEFUN ("x-get-resource", Fx_get_resource, Sx_get_resource, 2, 4, 0,
+  "Return the value of ATTRIBUTE, of class CLASS, from the X defaults database.\n\
+This uses `INSTANCE.ATTRIBUTE' as the key and `Emacs.CLASS' as the\n\
+class, where INSTANCE is the name under which Emacs was invoked, or\n\
+the name specified by the `-name' or `-rn' command-line arguments.\n\
+\n\
+The optional arguments COMPONENT and SUBCLASS add to the key and the\n\
+class, respectively.  You must specify both of them or neither.\n\
+If you specify them, the key is `INSTANCE.COMPONENT.ATTRIBUTE'\n\
+and the class is `Emacs.CLASS.SUBCLASS'.")
+  (attribute, class, component, subclass)
+     Lisp_Object attribute, class, component, subclass;
+{
+  register char *value;
+  char *name_key;
+  char *class_key;
+
+  CHECK_STRING (attribute, 0);
+  CHECK_STRING (class, 0);
+
+  if (!NILP (component))
+    CHECK_STRING (component, 1);
+  if (!NILP (subclass))
+    CHECK_STRING (subclass, 2);
+  if (NILP (component) != NILP (subclass))
+    error ("x-get-resource: must specify both COMPONENT and SUBCLASS or neither");
+
+  validate_x_resource_name ();
+
+  /* Allocate space for the components, the dots which separate them,
+     and the final '\0'.  Make them big enough for the worst case.  */
+  name_key = (char *) alloca (XSTRING (Vx_resource_name)->size
+			      + (STRINGP (component)
+				 ? XSTRING (component)->size : 0)
+			      + XSTRING (attribute)->size
+			      + 3);
+
+  class_key = (char *) alloca ((sizeof (EMACS_CLASS) - 1)
+			       + XSTRING (class)->size
+			       + (STRINGP (subclass)
+				  ? XSTRING (subclass)->size : 0)
+			       + 3);
+
+  /* Start with emacs.FRAMENAME for the name (the specific one)
+     and with `Emacs' for the class key (the general one).  */
+  strcpy (name_key, XSTRING (Vx_resource_name)->data);
+  strcpy (class_key, EMACS_CLASS);
+
+  strcat (class_key, ".");
+  strcat (class_key, XSTRING (class)->data);
+
+  if (!NILP (component))
+    {
+      strcat (class_key, ".");
+      strcat (class_key, XSTRING (subclass)->data);
+
+      strcat (name_key, ".");
+      strcat (name_key, XSTRING (component)->data);
+    }
+
+  strcat (name_key, ".");
+  strcat (name_key, XSTRING (attribute)->data);
+
+  value = x_get_string_resource (Qnil,
+				 name_key, class_key);
+
+  if (value != (char *) 0)
+    return build_string (value);
+  else
+    return Qnil;
+}
+
+/* Used when C code wants a resource value.  */
+
+char *
+x_get_resource_string (attribute, class)
+     char *attribute, *class;
+{
+  register char *value;
+  char *name_key;
+  char *class_key;
+
+  /* Allocate space for the components, the dots which separate them,
+     and the final '\0'.  */
+  name_key = (char *) alloca (XSTRING (Vinvocation_name)->size
+			      + strlen (attribute) + 2);
+  class_key = (char *) alloca ((sizeof (EMACS_CLASS) - 1)
+			       + strlen (class) + 2);
+
+  sprintf (name_key, "%s.%s",
+	   XSTRING (Vinvocation_name)->data,
+	   attribute);
+  sprintf (class_key, "%s.%s", EMACS_CLASS, class);
+
+  return x_get_string_resource (selected_frame,
+				name_key, class_key);
+}
+
+/* Types we might convert a resource string into.  */
+enum resource_types
+  {
+    number, boolean, string, symbol
+  };
+
+/* Return the value of parameter PARAM.
+
+   First search ALIST, then Vdefault_frame_alist, then the X defaults
+   database, using ATTRIBUTE as the attribute name and CLASS as its class.
+
+   Convert the resource to the type specified by desired_type.
+
+   If no default is specified, return Qunbound.  If you call
+   x_get_arg, make sure you deal with Qunbound in a reasonable way,
+   and don't let it get stored in any Lisp-visible variables!  */
+
+static Lisp_Object
+x_get_arg (alist, param, attribute, class, type)
+     Lisp_Object alist, param;
+     char *attribute;
+     char *class;
+     enum resource_types type;
+{
+  register Lisp_Object tem;
+
+  tem = Fassq (param, alist);
+  if (EQ (tem, Qnil))
+    tem = Fassq (param, Vdefault_frame_alist);
+  if (EQ (tem, Qnil))
+    {
+
+      if (attribute)
+	{
+	  tem = Fx_get_resource (build_string (attribute),
+				 build_string (class),
+				 Qnil, Qnil);
+
+	  if (NILP (tem))
+	    return Qunbound;
+
+	  switch (type)
+	    {
+	    case number:
+	      return make_number (atoi (XSTRING (tem)->data));
+
+	    case boolean:
+	      tem = Fdowncase (tem);
+	      if (!strcmp (XSTRING (tem)->data, "on")
+		  || !strcmp (XSTRING (tem)->data, "true"))
+		return Qt;
+	      else 
+		return Qnil;
+
+	    case string:
+	      return tem;
+
+	    case symbol:
+	      /* As a special case, we map the values `true' and `on'
+		 to Qt, and `false' and `off' to Qnil.  */
+	      {
+		Lisp_Object lower;
+		lower = Fdowncase (tem);
+		if (!strcmp (XSTRING (lower)->data, "on")
+		    || !strcmp (XSTRING (lower)->data, "true"))
+		  return Qt;
+		else if (!strcmp (XSTRING (lower)->data, "off")
+		      || !strcmp (XSTRING (lower)->data, "false"))
+		  return Qnil;
+		else
+		  return Fintern (tem, Qnil);
+	      }
+
+	    default:
+	      abort ();
+	    }
+	}
+      else
+	return Qunbound;
+    }
+  return Fcdr (tem);
+}
+
+/* Record in frame F the specified or default value according to ALIST
+   of the parameter named PARAM (a Lisp symbol).
+   If no value is specified for PARAM, look for an X default for XPROP
+   on the frame named NAME.
+   If that is not found either, use the value DEFLT.  */
+
+static Lisp_Object
+x_default_parameter (f, alist, prop, deflt, xprop, xclass, type)
+     struct frame *f;
+     Lisp_Object alist;
+     Lisp_Object prop;
+     Lisp_Object deflt;
+     char *xprop;
+     char *xclass;
+     enum resource_types type;
+{
+  Lisp_Object tem;
+
+  tem = x_get_arg (alist, prop, xprop, xclass, type);
+  if (EQ (tem, Qunbound))
+    tem = deflt;
+  x_set_frame_parameters (f, Fcons (Fcons (prop, tem), Qnil));
+  return tem;
+}
+
+DEFUN ("x-parse-geometry", Fx_parse_geometry, Sx_parse_geometry, 1, 1, 0,
+       "Parse an X-style geometry string STRING.\n\
+Returns an alist of the form ((top . TOP), (left . LEFT) ... ).\n\
+The properties returned may include `top', `left', `height', and `width'.\n\
+The value of `left' or `top' may be an integer,\n\
+or a list (+ N) meaning N pixels relative to top/left corner,\n\
+or a list (- N) meaning -N pixels relative to bottom/right corner.")
+     (string)
+     Lisp_Object string;
+{
+  int geometry, x, y;
+  unsigned int width, height;
+  Lisp_Object result;
+
+  CHECK_STRING (string, 0);
+
+  geometry = XParseGeometry ((char *) XSTRING (string)->data,
+			     &x, &y, &width, &height);
+
+  result = Qnil;
+  if (geometry & XValue)
+    {
+      Lisp_Object element;
+
+      if (x >= 0 && (geometry & XNegative))
+	element = Fcons (Qleft, Fcons (Qminus, Fcons (make_number (-x), Qnil)));
+      else if (x < 0 && ! (geometry & XNegative))
+	element = Fcons (Qleft, Fcons (Qplus, Fcons (make_number (x), Qnil)));
+      else
+	element = Fcons (Qleft, make_number (x));
+      result = Fcons (element, result);
+    }
+
+  if (geometry & YValue)
+    {
+      Lisp_Object element;
+
+      if (y >= 0 && (geometry & YNegative))
+	element = Fcons (Qtop, Fcons (Qminus, Fcons (make_number (-y), Qnil)));
+      else if (y < 0 && ! (geometry & YNegative))
+	element = Fcons (Qtop, Fcons (Qplus, Fcons (make_number (y), Qnil)));
+      else
+	element = Fcons (Qtop, make_number (y));
+      result = Fcons (element, result);
+    }
+
+  if (geometry & WidthValue)
+    result = Fcons (Fcons (Qwidth, make_number (width)), result);
+  if (geometry & HeightValue)
+    result = Fcons (Fcons (Qheight, make_number (height)), result);
+
+  return result;
+}
+
+/* Calculate the desired size and position of this window,
+   and return the flags saying which aspects were specified.
+
+   This function does not make the coordinates positive.  */
+
+#define DEFAULT_ROWS 40
+#define DEFAULT_COLS 80
+
+static int
+x_figure_window_size (f, parms)
+     struct frame *f;
+     Lisp_Object parms;
+{
+  register Lisp_Object tem0, tem1, tem2;
+  int height, width, left, top;
+  register int geometry;
+  long window_prompting = 0;
+
+  /* Default values if we fall through.
+     Actually, if that happens we should get
+     window manager prompting.  */
+  f->width = DEFAULT_COLS;
+  f->height = DEFAULT_ROWS;
+  /* Window managers expect that if program-specified
+     positions are not (0,0), they're intentional, not defaults.  */
+  f->output_data.win32->top_pos = 0;
+  f->output_data.win32->left_pos = 0;
+
+  tem0 = x_get_arg (parms, Qheight, 0, 0, number);
+  tem1 = x_get_arg (parms, Qwidth, 0, 0, number);
+  tem2 = x_get_arg (parms, Quser_size, 0, 0, number);
+  if (! EQ (tem0, Qunbound) || ! EQ (tem1, Qunbound))
+    {
+      if (!EQ (tem0, Qunbound))
+	{
+	  CHECK_NUMBER (tem0, 0);
+	  f->height = XINT (tem0);
+	}
+      if (!EQ (tem1, Qunbound))
+	{
+	  CHECK_NUMBER (tem1, 0);
+	  f->width = XINT (tem1);
+	}
+      if (!NILP (tem2) && !EQ (tem2, Qunbound))
+	window_prompting |= USSize;
+      else
+	window_prompting |= PSize;
+    }
+
+  f->output_data.win32->vertical_scroll_bar_extra
+    = (!FRAME_HAS_VERTICAL_SCROLL_BARS (f)
+       ? 0
+       : FRAME_SCROLL_BAR_PIXEL_WIDTH (f) > 0
+       ? FRAME_SCROLL_BAR_PIXEL_WIDTH (f)
+       : (FRAME_SCROLL_BAR_COLS (f) * FONT_WIDTH (f->output_data.win32->font)));
+  f->output_data.win32->pixel_width = CHAR_TO_PIXEL_WIDTH (f, f->width);
+  f->output_data.win32->pixel_height = CHAR_TO_PIXEL_HEIGHT (f, f->height);
+
+  tem0 = x_get_arg (parms, Qtop, 0, 0, number);
+  tem1 = x_get_arg (parms, Qleft, 0, 0, number);
+  tem2 = x_get_arg (parms, Quser_position, 0, 0, number);
+  if (! EQ (tem0, Qunbound) || ! EQ (tem1, Qunbound))
+    {
+      if (EQ (tem0, Qminus))
+	{
+	  f->output_data.win32->top_pos = 0;
+	  window_prompting |= YNegative;
+	}
+      else if (CONSP (tem0) && EQ (XCONS (tem0)->car, Qminus)
+	       && CONSP (XCONS (tem0)->cdr)
+	       && INTEGERP (XCONS (XCONS (tem0)->cdr)->car))
+	{
+	  f->output_data.win32->top_pos = - XINT (XCONS (XCONS (tem0)->cdr)->car);
+	  window_prompting |= YNegative;
+	}
+      else if (CONSP (tem0) && EQ (XCONS (tem0)->car, Qplus)
+	       && CONSP (XCONS (tem0)->cdr)
+	       && INTEGERP (XCONS (XCONS (tem0)->cdr)->car))
+	{
+	  f->output_data.win32->top_pos = XINT (XCONS (XCONS (tem0)->cdr)->car);
+	}
+      else if (EQ (tem0, Qunbound))
+	f->output_data.win32->top_pos = 0;
+      else
+	{
+	  CHECK_NUMBER (tem0, 0);
+	  f->output_data.win32->top_pos = XINT (tem0);
+	  if (f->output_data.win32->top_pos < 0)
+	    window_prompting |= YNegative;
+	}
+
+      if (EQ (tem1, Qminus))
+	{
+	  f->output_data.win32->left_pos = 0;
+	  window_prompting |= XNegative;
+	}
+      else if (CONSP (tem1) && EQ (XCONS (tem1)->car, Qminus)
+	       && CONSP (XCONS (tem1)->cdr)
+	       && INTEGERP (XCONS (XCONS (tem1)->cdr)->car))
+	{
+	  f->output_data.win32->left_pos = - XINT (XCONS (XCONS (tem1)->cdr)->car);
+	  window_prompting |= XNegative;
+	}
+      else if (CONSP (tem1) && EQ (XCONS (tem1)->car, Qplus)
+	       && CONSP (XCONS (tem1)->cdr)
+	       && INTEGERP (XCONS (XCONS (tem1)->cdr)->car))
+	{
+	  f->output_data.win32->left_pos = XINT (XCONS (XCONS (tem1)->cdr)->car);
+	}
+      else if (EQ (tem1, Qunbound))
+	f->output_data.win32->left_pos = 0;
+      else
+	{
+	  CHECK_NUMBER (tem1, 0);
+	  f->output_data.win32->left_pos = XINT (tem1);
+	  if (f->output_data.win32->left_pos < 0)
+	    window_prompting |= XNegative;
+	}
+
+      if (!NILP (tem2) && ! EQ (tem2, Qunbound))
+	window_prompting |= USPosition;
+      else
+	window_prompting |= PPosition;
+    }
+
+  return window_prompting;
+}
+
+
+
+extern LRESULT CALLBACK win32_wnd_proc ();
+
+BOOL 
+win32_init_class (hinst)
+     HINSTANCE hinst;
+{
+  WNDCLASS wc;
+
+  wc.style = CS_HREDRAW | CS_VREDRAW | CS_OWNDC;
+  wc.lpfnWndProc = (WNDPROC) win32_wnd_proc;
+  wc.cbClsExtra = 0;
+  wc.cbWndExtra = WND_EXTRA_BYTES;
+  wc.hInstance = hinst;
+  wc.hIcon = LoadIcon (hinst, EMACS_CLASS);
+  wc.hCursor = LoadCursor (NULL, IDC_ARROW);
+  wc.hbrBackground = NULL; // GetStockObject (WHITE_BRUSH);
+  wc.lpszMenuName = NULL;
+  wc.lpszClassName = EMACS_CLASS;
+
+  return (RegisterClass (&wc));
+}
+
+HWND 
+win32_createscrollbar (f, bar)
+     struct frame *f;
+     struct scroll_bar * bar;
+{
+  return (CreateWindow ("SCROLLBAR", "", SBS_VERT | WS_CHILD | WS_VISIBLE,
+			/* Position and size of scroll bar.  */
+			XINT(bar->left), XINT(bar->top), 
+			XINT(bar->width), XINT(bar->height),
+			FRAME_WIN32_WINDOW (f),
+			NULL,
+			hinst,
+			NULL));
+}
+
+void 
+win32_createwindow (f)
+     struct frame *f;
+{
+  HWND hwnd;
+  
+  /* Do first time app init */
+  
+  if (!hprevinst)
+    {
+      win32_init_class (hinst);
+    }
+  
+  FRAME_WIN32_WINDOW (f) = hwnd = CreateWindow (EMACS_CLASS,
+						f->namebuf,
+						f->output_data.win32->dwStyle | WS_CLIPCHILDREN,
+						f->output_data.win32->left_pos,
+						f->output_data.win32->top_pos,
+						PIXEL_WIDTH (f),
+						PIXEL_HEIGHT (f),
+						NULL,
+						NULL,
+						hinst,
+						NULL);
+  
+  if (hwnd)
+    {
+      SetWindowLong (hwnd, WND_X_UNITS_INDEX, FONT_WIDTH (f->output_data.win32->font));
+      SetWindowLong (hwnd, WND_Y_UNITS_INDEX, f->output_data.win32->line_height);
+      SetWindowLong (hwnd, WND_BACKGROUND_INDEX, f->output_data.win32->background_pixel);
+    }
+}
+
+DWORD 
+win_msg_worker (dw)
+     DWORD dw;
+{
+  MSG msg;
+  
+  /* Ensure our message queue is created */
+  
+  PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE);
+  
+  PostThreadMessage (dwMainThreadId, WM_EMACS_DONE, 0, 0);
+  
+  while (GetMessage (&msg, NULL, 0, 0))
+    {
+      if (msg.hwnd == NULL)
+	{
+	  switch (msg.message)
+	    {
+	    case WM_EMACS_CREATEWINDOW:
+	      win32_createwindow ((struct frame *) msg.wParam);
+	      PostThreadMessage (dwMainThreadId, WM_EMACS_DONE, 0, 0);
+	      break;
+	    case WM_EMACS_CREATESCROLLBAR:
+	      {
+		HWND hwnd = win32_createscrollbar ((struct frame *) msg.wParam,
+						   (struct scroll_bar *) msg.lParam);
+		PostThreadMessage (dwMainThreadId, WM_EMACS_DONE, (WPARAM)hwnd, 0);
+	      }
+	      break;
+	    case WM_EMACS_KILL:
+	      return (0);
+	    }
+	}
+      else
+	{
+	  DispatchMessage (&msg);
+	}
+    }
+  
+  return (0);
+}
+
+HDC 
+map_mode (hdc)
+     HDC hdc;
+{
+  if (hdc) 
+    {
+#if 0
+      /* Make mapping mode be in 1/20 of point */
+      
+      SetMapMode (hdc, MM_ANISOTROPIC);
+      SetWindowExtEx (hdc, 1440, 1440, NULL);
+      SetViewportExtEx (hdc,
+			GetDeviceCaps (hdc, LOGPIXELSX),
+			GetDeviceCaps (hdc, LOGPIXELSY),
+			NULL);
+#endif
+    }
+  return (hdc);
+}
+
+/* Convert between the modifier bits Win32 uses and the modifier bits
+   Emacs uses.  */
+unsigned int
+win32_get_modifiers ()
+{
+  return (((GetKeyState (VK_SHIFT)&0x8000)   ? shift_modifier  : 0) |
+	  ((GetKeyState (VK_CONTROL)&0x8000) ? ctrl_modifier   : 0) |
+	  ((GetKeyState (VK_MENU)&0x8000)    ? meta_modifier   : 0));
+}
+
+void 
+my_post_msg (wmsg, hwnd, msg, wParam, lParam)
+     Win32Msg * wmsg;
+     HWND hwnd;
+     UINT msg;
+     WPARAM wParam;
+     LPARAM lParam;
+{
+  wmsg->msg.hwnd = hwnd;
+  wmsg->msg.message = msg;
+  wmsg->msg.wParam = wParam;
+  wmsg->msg.lParam = lParam;
+  wmsg->msg.time = GetMessageTime ();
+
+  post_msg (wmsg);
+}
+
+/* Main window procedure */
+
+extern char *lispy_function_keys[];
+
+LRESULT CALLBACK 
+win32_wnd_proc (hwnd, msg, wParam, lParam)
+     HWND hwnd;
+     UINT msg;
+     WPARAM wParam;
+     LPARAM lParam;
+{
+  struct frame *f;
+  LRESULT ret = 1;
+  struct win32_display_info *dpyinfo = &one_win32_display_info;
+  Win32Msg wmsg;
+  
+  switch (msg) 
+    {
+    case WM_ERASEBKGND:
+      {
+	HBRUSH hb;
+	HANDLE oldobj;
+	RECT rect;
+	
+	GetClientRect (hwnd, &rect);
+	
+	hb = CreateSolidBrush (GetWindowLong (hwnd, WND_BACKGROUND_INDEX));
+	
+	oldobj = SelectObject ((HDC)wParam, hb);
+	
+	FillRect((HDC)wParam, &rect, hb);
+	
+	SelectObject((HDC)wParam, oldobj);
+	
+	DeleteObject (hb);
+	
+	return (0);
+      }
+    case WM_PAINT:
+      {
+	PAINTSTRUCT paintStruct;
+		    
+	BeginPaint (hwnd, &paintStruct);
+	wmsg.rect = paintStruct.rcPaint;
+	EndPaint (hwnd, &paintStruct);
+      
+	my_post_msg (&wmsg, hwnd, msg, wParam, lParam);
+      
+	return (0);
+      }
+      
+    case WM_CREATE:
+      {
+	HDC hdc = my_get_dc (hwnd);
+
+	/* Make mapping mode be in 1/20 of point */
+
+	map_mode (hdc);
+
+	ReleaseDC (hwnd, hdc);
+      }
+      
+      return (0);
+    case WM_KEYDOWN:
+    case WM_SYSKEYDOWN:
+#if 0
+      if (! ((wParam >= VK_BACK && wParam <= VK_TAB)
+	     || (wParam >= VK_CLEAR && wParam <= VK_RETURN)
+	     || (wParam == VK_ESCAPE)
+	     || (wParam >= VK_PRIOR && wParam <= VK_HELP)
+	     || (wParam >= VK_LWIN && wParam <= VK_APPS)
+	     || (wParam >= VK_NUMPAD0 && wParam <= VK_F24)
+	     || (wParam >= VK_NUMLOCK && wParam <= VK_SCROLL)
+	     || (wParam >= VK_ATTN && wParam <= VK_OEM_CLEAR)
+	     || !TranslateMessage (&msg1)))
+	{
+	  goto dflt;
+	}
+#endif
+      
+      /* Check for special characters since translate message 
+	 seems to always indicate true.  */
+      
+      if (wParam == VK_MENU
+	  || wParam == VK_SHIFT
+	  || wParam == VK_CONTROL
+	  || wParam == VK_CAPITAL)
+	break;
+      
+      /* Anything we do not have a name for needs to be translated or 
+	 returned as ascii keystroke.  */
+      
+      if (lispy_function_keys[wParam] == 0)
+	{
+	  MSG msg1;
+	    
+	  msg1.hwnd = hwnd;
+	  msg1.message = msg;
+	  msg1.wParam = wParam;
+	  msg1.lParam = lParam;
+	    
+	  if (TranslateMessage (&msg1))
+	    break;
+	  else
+	    msg = WM_CHAR;
+	}
+      
+      /* Fall through */
+      
+    case WM_SYSCHAR:
+    case WM_CHAR:
+      wmsg.dwModifiers = win32_get_modifiers ();
+      
+      my_post_msg (&wmsg, hwnd, msg, wParam, lParam);
+      break;
+    case WM_LBUTTONDOWN:
+    case WM_LBUTTONUP:
+    case WM_MBUTTONDOWN:
+    case WM_MBUTTONUP:
+    case WM_RBUTTONDOWN:
+    case WM_RBUTTONUP:
+      {
+	BOOL up;
+
+	if (parse_button (msg, NULL, &up))
+	  {
+	    if (up) ReleaseCapture ();
+	    else SetCapture (hwnd);
+	  }
+      }
+      
+      wmsg.dwModifiers = win32_get_modifiers ();
+      
+      my_post_msg (&wmsg, hwnd, msg, wParam, lParam);
+      goto dflt;
+    case WM_MOUSEMOVE:
+    case WM_MOVE:
+    case WM_SIZE:
+    case WM_SETFOCUS:
+    case WM_KILLFOCUS:
+    case WM_CLOSE:
+    case WM_VSCROLL:
+    case WM_SYSCOMMAND:
+    case WM_COMMAND:
+      my_post_msg (&wmsg, hwnd, msg, wParam, lParam);
+      goto dflt;
+    case WM_WINDOWPOSCHANGING:
+      {
+	WINDOWPLACEMENT wp;
+	LPWINDOWPOS lppos = (WINDOWPOS *) lParam;
+	
+	GetWindowPlacement (hwnd, &wp);
+	
+	if (wp.showCmd != SW_SHOWMINIMIZED && ! (lppos->flags & SWP_NOSIZE))
+	  {
+	    RECT rect;
+	    int wdiff;
+	    int hdiff;
+	    DWORD dwXUnits;
+	    DWORD dwYUnits;
+	    RECT wr;
+	    
+	    GetWindowRect (hwnd, &wr);
+	    
+	    enter_crit ();
+	    
+	    dwXUnits = GetWindowLong (hwnd, WND_X_UNITS_INDEX);
+	    dwYUnits = GetWindowLong (hwnd, WND_Y_UNITS_INDEX);
+	    
+	    leave_crit ();
+	    
+	    memset (&rect, 0, sizeof (rect));
+	    AdjustWindowRect (&rect, GetWindowLong (hwnd, GWL_STYLE), 
+			      GetMenu (hwnd) != NULL);
+
+	    /* All windows have an extra pixel so subtract 1 */
+	    
+	    wdiff = (lppos->cx - (rect.right - rect.left) - 0) % dwXUnits;
+	    hdiff = (lppos->cy - (rect.bottom - rect.top) - 0) % dwYUnits;
+	    
+	    if (wdiff || hdiff)
+	      {
+		/* For right/bottom sizing we can just fix the sizes.  
+		   However for top/left sizing we will need to fix the X 
+		   and Y positions as well.  */
+		
+		lppos->cx -= wdiff;
+		lppos->cy -= hdiff;
+		
+		if (wp.showCmd != SW_SHOWMAXIMIZED 
+		    && ! (lppos->flags & SWP_NOMOVE))
+		  {
+		    if (lppos->x != wr.left || lppos->y != wr.top)
+		      {
+			lppos->x += wdiff;
+			lppos->y += hdiff;
+		      }
+		    else
+		      {
+			lppos->flags |= SWP_NOMOVE;
+		      }
+		  }
+		
+		ret = 0;
+	      }
+	  }
+      }
+    
+      if (ret == 0) return (0);
+      
+      goto dflt;
+    case WM_EMACS_DESTROYWINDOW:
+      DestroyWindow ((HWND) wParam);
+      break;
+    default:
+    dflt:
+      return DefWindowProc (hwnd, msg, wParam, lParam);
+    }
+  
+  return (1);
+}
+
+void 
+my_create_window (f)
+     struct frame * f;
+{
+  MSG msg;
+
+  PostThreadMessage (dwWinThreadId, WM_EMACS_CREATEWINDOW, (WPARAM)f, 0);
+  GetMessage (&msg, NULL, WM_EMACS_DONE, WM_EMACS_DONE);
+}
+
+/* Create and set up the win32 window for frame F.  */
+
+static void
+win32_window (f, window_prompting, minibuffer_only)
+     struct frame *f;
+     long window_prompting;
+     int minibuffer_only;
+{
+  BLOCK_INPUT;
+
+  /* Use the resource name as the top-level window name
+     for looking up resources.  Make a non-Lisp copy
+     for the window manager, so GC relocation won't bother it.
+
+     Elsewhere we specify the window name for the window manager.  */
+     
+  {
+    char *str = (char *) XSTRING (Vx_resource_name)->data;
+    f->namebuf = (char *) xmalloc (strlen (str) + 1);
+    strcpy (f->namebuf, str);
+  }
+
+  my_create_window (f);
+
+  validate_x_resource_name ();
+
+  /* x_set_name normally ignores requests to set the name if the
+     requested name is the same as the current name.  This is the one
+     place where that assumption isn't correct; f->name is set, but
+     the server hasn't been told.  */
+  {
+    Lisp_Object name;
+    int explicit = f->explicit_name;
+
+    f->explicit_name = 0;
+    name = f->name;
+    f->name = Qnil;
+    x_set_name (f, name, explicit);
+  }
+
+  UNBLOCK_INPUT;
+
+  if (!minibuffer_only && FRAME_EXTERNAL_MENU_BAR (f))
+    initialize_frame_menubar (f);
+
+  if (FRAME_WIN32_WINDOW (f) == 0)
+    error ("Unable to create window");
+}
+
+/* Handle the icon stuff for this window.  Perhaps later we might
+   want an x_set_icon_position which can be called interactively as
+   well.  */
+
+static void
+x_icon (f, parms)
+     struct frame *f;
+     Lisp_Object parms;
+{
+  Lisp_Object icon_x, icon_y;
+
+  /* Set the position of the icon.  Note that win95 groups all
+     icons in the tray.  */
+  icon_x = x_get_arg (parms, Qicon_left, 0, 0, number);
+  icon_y = x_get_arg (parms, Qicon_top, 0, 0, number);
+  if (!EQ (icon_x, Qunbound) && !EQ (icon_y, Qunbound))
+    {
+      CHECK_NUMBER (icon_x, 0);
+      CHECK_NUMBER (icon_y, 0);
+    }
+  else if (!EQ (icon_x, Qunbound) || !EQ (icon_y, Qunbound))
+    error ("Both left and top icon corners of icon must be specified");
+
+  BLOCK_INPUT;
+
+  if (! EQ (icon_x, Qunbound))
+    x_wm_set_icon_position (f, XINT (icon_x), XINT (icon_y));
+
+  UNBLOCK_INPUT;
+}
+
+DEFUN ("x-create-frame", Fx_create_frame, Sx_create_frame,
+       1, 1, 0,
+  "Make a new window, which is called a \"frame\" in Emacs terms.\n\
+Returns an Emacs frame object.\n\
+ALIST is an alist of frame parameters.\n\
+If the parameters specify that the frame should not have a minibuffer,\n\
+and do not specify a specific minibuffer window to use,\n\
+then `default-minibuffer-frame' must be a frame whose minibuffer can\n\
+be shared by the new frame.\n\
+\n\
+This function is an internal primitive--use `make-frame' instead.")
+  (parms)
+     Lisp_Object parms;
+{
+  struct frame *f;
+  Lisp_Object frame, tem;
+  Lisp_Object name;
+  int minibuffer_only = 0;
+  long window_prompting = 0;
+  int width, height;
+  int count = specpdl_ptr - specpdl;
+  struct gcpro gcpro1;
+  Lisp_Object display;
+  struct win32_display_info *dpyinfo;
+  Lisp_Object parent;
+  struct kboard *kb;
+
+  /* Use this general default value to start with
+     until we know if this frame has a specified name.  */
+  Vx_resource_name = Vinvocation_name;
+
+  display = x_get_arg (parms, Qdisplay, 0, 0, string);
+  if (EQ (display, Qunbound))
+    display = Qnil;
+  dpyinfo = check_x_display_info (display);
+#ifdef MULTI_KBOARD
+  kb = dpyinfo->kboard;
+#else
+  kb = &the_only_kboard;
+#endif
+
+  name = x_get_arg (parms, Qname, "title", "Title", string);
+  if (!STRINGP (name)
+      && ! EQ (name, Qunbound)
+      && ! NILP (name))
+    error ("Invalid frame name--not a string or nil");
+
+  if (STRINGP (name))
+    Vx_resource_name = name;
+
+  /* See if parent window is specified.  */
+  parent = x_get_arg (parms, Qparent_id, NULL, NULL, number);
+  if (EQ (parent, Qunbound))
+    parent = Qnil;
+  if (! NILP (parent))
+    CHECK_NUMBER (parent, 0);
+
+  tem = x_get_arg (parms, Qminibuffer, 0, 0, symbol);
+  if (EQ (tem, Qnone) || NILP (tem))
+    f = make_frame_without_minibuffer (Qnil, kb, display);
+  else if (EQ (tem, Qonly))
+    {
+      f = make_minibuffer_frame ();
+      minibuffer_only = 1;
+    }
+  else if (WINDOWP (tem))
+    f = make_frame_without_minibuffer (tem, kb, display);
+  else
+    f = make_frame (1);
+
+  /* Note that Windows does support scroll bars.  */
+  FRAME_CAN_HAVE_SCROLL_BARS (f) = 1;
+
+  XSETFRAME (frame, f);
+  GCPRO1 (frame);
+
+  f->output_method = output_win32;
+  f->output_data.win32 = (struct win32_output *) xmalloc (sizeof (struct win32_output));
+  bzero (f->output_data.win32, sizeof (struct win32_output));
+
+/*  FRAME_WIN32_DISPLAY_INFO (f) = dpyinfo; */
+#ifdef MULTI_KBOARD
+  FRAME_KBOARD (f) = kb;
+#endif
+
+  /* Specify the parent under which to make this window.  */
+
+  if (!NILP (parent))
+    {
+      f->output_data.win32->parent_desc = (Window) parent;
+      f->output_data.win32->explicit_parent = 1;
+    }
+  else
+    {
+      f->output_data.win32->parent_desc = FRAME_WIN32_DISPLAY_INFO (f)->root_window;
+      f->output_data.win32->explicit_parent = 0;
+    }
+
+  /* Note that the frame has no physical cursor right now.  */
+  f->phys_cursor_x = -1;
+
+  /* Set the name; the functions to which we pass f expect the name to
+     be set.  */
+  if (EQ (name, Qunbound) || NILP (name))
+    {
+      f->name = build_string (dpyinfo->win32_id_name);
+      f->explicit_name = 0;
+    }
+  else
+    {
+      f->name = name;
+      f->explicit_name = 1;
+      /* use the frame's title when getting resources for this frame.  */
+      specbind (Qx_resource_name, name);
+    }
+
+  /* Extract the window parameters from the supplied values
+     that are needed to determine window geometry.  */
+  {
+    Lisp_Object font;
+
+    font = x_get_arg (parms, Qfont, "font", "Font", string);
+    BLOCK_INPUT;
+    /* First, try whatever font the caller has specified.  */
+    if (STRINGP (font))
+      font = x_new_font (f, XSTRING (font)->data);
+#if 0
+    /* Try out a font which we hope has bold and italic variations.  */
+    if (!STRINGP (font))
+      font = x_new_font (f, "-misc-fixed-medium-r-normal-*-*-140-*-*-c-*-iso8859-1");
+    if (! STRINGP (font))
+      font = x_new_font (f, "-*-*-medium-r-normal-*-*-140-*-*-c-*-iso8859-1");
+    if (! STRINGP (font))
+      /* This was formerly the first thing tried, but it finds too many fonts
+	 and takes too long.  */
+      font = x_new_font (f, "-*-*-medium-r-*-*-*-*-*-*-c-*-iso8859-1");
+    /* If those didn't work, look for something which will at least work.  */
+    if (! STRINGP (font))
+      font = x_new_font (f, "-*-fixed-*-*-*-*-*-140-*-*-c-*-iso8859-1");
+    if (! STRINGP (font))
+      font = x_new_font (f, "-*-system-medium-r-normal-*-*-200-*-*-c-120-*-*");
+#endif
+    if (! STRINGP (font))
+      font = x_new_font (f, "-*-terminal-medium-r-normal-*-*-180-*-*-c-120-*-*");
+    UNBLOCK_INPUT;
+    if (! STRINGP (font))
+      font = build_string ("-*-system");
+
+    x_default_parameter (f, parms, Qfont, font, 
+			 "font", "Font", string);
+  }
+
+  x_default_parameter (f, parms, Qborder_width, make_number (2),
+		       "borderwidth", "BorderWidth", number);
+  /* This defaults to 2 in order to match xterm.  We recognize either
+     internalBorderWidth or internalBorder (which is what xterm calls
+     it).  */
+  if (NILP (Fassq (Qinternal_border_width, parms)))
+    {
+      Lisp_Object value;
+
+      value = x_get_arg (parms, Qinternal_border_width,
+			 "internalBorder", "BorderWidth", number);
+      if (! EQ (value, Qunbound))
+	parms = Fcons (Fcons (Qinternal_border_width, value),
+		       parms);
+    }
+  x_default_parameter (f, parms, Qinternal_border_width, make_number (0),
+		       "internalBorderWidth", "BorderWidth", number);
+  x_default_parameter (f, parms, Qvertical_scroll_bars, Qt,
+		       "verticalScrollBars", "ScrollBars", boolean);
+
+  /* Also do the stuff which must be set before the window exists.  */
+  x_default_parameter (f, parms, Qforeground_color, build_string ("black"),
+		       "foreground", "Foreground", string);
+  x_default_parameter (f, parms, Qbackground_color, build_string ("white"),
+		       "background", "Background", string);
+  x_default_parameter (f, parms, Qmouse_color, build_string ("black"),
+		       "pointerColor", "Foreground", string);
+  x_default_parameter (f, parms, Qcursor_color, build_string ("black"),
+		       "cursorColor", "Foreground", string);
+  x_default_parameter (f, parms, Qborder_color, build_string ("black"),
+		       "borderColor", "BorderColor", string);
+
+  x_default_parameter (f, parms, Qmenu_bar_lines, make_number (1),
+		       "menuBar", "MenuBar", number);
+  x_default_parameter (f, parms, Qscroll_bar_width, Qnil,
+		       "scrollBarWidth", "ScrollBarWidth", number);
+
+  f->output_data.win32->dwStyle = WS_OVERLAPPEDWINDOW;
+  f->output_data.win32->parent_desc = FRAME_WIN32_DISPLAY_INFO (f)->root_window;
+  window_prompting = x_figure_window_size (f, parms);
+
+  if (window_prompting & XNegative)
+    {
+      if (window_prompting & YNegative)
+	f->output_data.win32->win_gravity = SouthEastGravity;
+      else
+	f->output_data.win32->win_gravity = NorthEastGravity;
+    }
+  else
+    {
+      if (window_prompting & YNegative)
+	f->output_data.win32->win_gravity = SouthWestGravity;
+      else
+	f->output_data.win32->win_gravity = NorthWestGravity;
+    }
+
+  f->output_data.win32->size_hint_flags = window_prompting;
+
+  win32_window (f, window_prompting, minibuffer_only);
+  x_icon (f, parms);
+  init_frame_faces (f);
+
+  /* We need to do this after creating the window, so that the
+     icon-creation functions can say whose icon they're describing.  */
+  x_default_parameter (f, parms, Qicon_type, Qnil,
+		       "bitmapIcon", "BitmapIcon", symbol);
+
+  x_default_parameter (f, parms, Qauto_raise, Qnil,
+		       "autoRaise", "AutoRaiseLower", boolean);
+  x_default_parameter (f, parms, Qauto_lower, Qnil,
+		       "autoLower", "AutoRaiseLower", boolean);
+  x_default_parameter (f, parms, Qcursor_type, Qbox,
+		       "cursorType", "CursorType", symbol);
+
+  /* Dimensions, especially f->height, must be done via change_frame_size.
+     Change will not be effected unless different from the current
+     f->height.  */
+  width = f->width;
+  height = f->height;
+  f->height = f->width = 0;
+  change_frame_size (f, height, width, 1, 0);
+
+  /* Tell the server what size and position, etc, we want,
+     and how badly we want them.  */
+  BLOCK_INPUT;
+  x_wm_set_size_hint (f, window_prompting, 0);
+  UNBLOCK_INPUT;
+
+  tem = x_get_arg (parms, Qunsplittable, 0, 0, boolean);
+  f->no_split = minibuffer_only || EQ (tem, Qt);
+
+  UNGCPRO;
+
+  /* It is now ok to make the frame official
+     even if we get an error below.
+     And the frame needs to be on Vframe_list
+     or making it visible won't work.  */
+  Vframe_list = Fcons (frame, Vframe_list);
+
+  /* Now that the frame is official, it counts as a reference to
+     its display.  */
+  FRAME_WIN32_DISPLAY_INFO (f)->reference_count++;
+
+  /* Make the window appear on the frame and enable display,
+     unless the caller says not to.  However, with explicit parent,
+     Emacs cannot control visibility, so don't try.  */
+  if (! f->output_data.win32->explicit_parent)
+    {
+      Lisp_Object visibility;
+
+      visibility = x_get_arg (parms, Qvisibility, 0, 0, symbol);
+      if (EQ (visibility, Qunbound))
+	visibility = Qt;
+
+      if (EQ (visibility, Qicon))
+	x_iconify_frame (f);
+      else if (! NILP (visibility))
+	x_make_frame_visible (f);
+      else
+	/* Must have been Qnil.  */
+	;
+    }
+
+  return unbind_to (count, frame);
+}
+
+/* FRAME is used only to get a handle on the X display.  We don't pass the
+   display info directly because we're called from frame.c, which doesn't
+   know about that structure.  */
+Lisp_Object
+x_get_focus_frame (frame)
+     struct frame *frame;
+{
+  struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (frame);
+  Lisp_Object xfocus;
+  if (! dpyinfo->win32_focus_frame)
+    return Qnil;
+
+  XSETFRAME (xfocus, dpyinfo->win32_focus_frame);
+  return xfocus;
+}
+
+DEFUN ("focus-frame", Ffocus_frame, Sfocus_frame, 1, 1, 0,
+  "Set the focus on FRAME.")
+  (frame)
+     Lisp_Object frame;
+{
+  CHECK_LIVE_FRAME (frame, 0);
+
+  if (FRAME_WIN32_P (XFRAME (frame)))
+    {
+      BLOCK_INPUT;
+      x_focus_on_frame (XFRAME (frame));
+      UNBLOCK_INPUT;
+      return frame;
+    }
+
+  return Qnil;
+}
+
+DEFUN ("unfocus-frame", Funfocus_frame, Sunfocus_frame, 0, 0, 0,
+       "If a frame has been focused, release it.")
+     ()
+{
+  if (FRAME_WIN32_P (selected_frame))
+    {
+      struct win32_display_info *dpyinfo = FRAME_WIN32_DISPLAY_INFO (selected_frame);
+
+      if (dpyinfo->win32_focus_frame)
+	{
+	  BLOCK_INPUT;
+	  x_unfocus_frame (dpyinfo->win32_focus_frame);
+	  UNBLOCK_INPUT;
+	}
+    }
+  
+  return Qnil;
+}
+
+XFontStruct 
+*win32_load_font (dpyinfo,name)
+struct win32_display_info *dpyinfo;
+char * name;
+{
+  XFontStruct * font = NULL;
+  BOOL ok;
+  
+  {
+    LOGFONT lf;
+	
+    if (!name || !x_to_win32_font(name, &lf)) 
+      return (NULL);
+	
+    font = (XFontStruct *) xmalloc (sizeof (XFontStruct));
+	
+    if (!font) return (NULL);
+	
+    BLOCK_INPUT;
+	
+    font->hfont = CreateFontIndirect(&lf);
+  }
+  
+    if (font->hfont == NULL) 
+      {
+	ok = FALSE;
+      } 
+    else 
+      {
+	HDC hdc;
+	HANDLE oldobj;
+
+	hdc = my_get_dc (dpyinfo->root_window);
+	
+	oldobj = SelectObject (hdc, font->hfont);
+	
+	ok = GetTextMetrics (hdc, &font->tm);
+	
+	SelectObject (hdc, oldobj);
+	
+	ReleaseDC (dpyinfo->root_window, hdc);
+      }
+  
+  UNBLOCK_INPUT;
+  
+  if (ok) return (font);
+  
+  win32_unload_font(dpyinfo, font);
+  return (NULL);
+}
+
+void 
+win32_unload_font (dpyinfo, font)
+     struct win32_display_info *dpyinfo;
+     XFontStruct * font;
+{
+  if (font) 
+    {
+      if (font->hfont) DeleteObject(font->hfont);
+      xfree (font);
+    }
+}
+
+/* The font conversion stuff between x and win32 */
+
+/* X font string is as follows (from faces.el)
+ * (let ((- 		"[-?]")
+ *      (foundry	"[^-]+")
+ *      (family 	"[^-]+")
+ *      (weight		"\\(bold\\|demibold\\|medium\\)")		; 1
+ *      (weight\?	"\\([^-]*\\)")					; 1
+ *      (slant		"\\([ior]\\)")					; 2
+ *      (slant\?	"\\([^-]?\\)")					; 2
+ *      (swidth		"\\([^-]*\\)")					; 3
+ *      (adstyle	"[^-]*")					; 4
+ *      (pixelsize	"[0-9]+")
+ *      (pointsize	"[0-9][0-9]+")
+ *      (resx		"[0-9][0-9]+")
+ *      (resy		"[0-9][0-9]+")
+ *      (spacing	"[cmp?*]")
+ *      (avgwidth	"[0-9]+")
+ *      (registry	"[^-]+")
+ *      (encoding	"[^-]+")
+ *      )
+ *  (setq x-font-regexp
+ *	(concat "\\`\\*?[-?*]"
+ *		foundry - family - weight\? - slant\? - swidth - adstyle -
+ *		pixelsize - pointsize - resx - resy - spacing - registry -
+ *		encoding "[-?*]\\*?\\'"
+ *		))
+ *  (setq x-font-regexp-head
+ *	(concat "\\`[-?*]" foundry - family - weight\? - slant\?
+ *		"\\([-*?]\\|\\'\\)"))
+ *  (setq x-font-regexp-slant (concat - slant -))
+ *  (setq x-font-regexp-weight (concat - weight -))
+ * nil)	    
+ */
+    
+#define FONT_START       "[-?]"
+#define FONT_FOUNDRY     "[^-]+"
+#define FONT_FAMILY      "\\([^-]+\\)"                      /* 1 */
+#define FONT_WEIGHT      "\\(bold\\|demibold\\|medium\\)"   /* 2 */
+#define FONT_WEIGHT_Q    "\\([^-]*\\)"                      /* 2 */
+#define FONT_SLANT       "\\([ior]\\)"                      /* 3 */
+#define FONT_SLANT_Q     "\\([^-]?\\)"                      /* 3 */
+#define FONT_SWIDTH      "\\([^-]*\\)"                      /* 4 */
+#define FONT_ADSTYLE     "[^-]*"
+#define FONT_PIXELSIZE   "[^-]*"
+#define FONT_POINTSIZE   "\\([0-9][0-9]+\\|\\*\\)"          /* 5 */
+#define FONT_RESX        "[0-9][0-9]+"
+#define FONT_RESY        "[0-9][0-9]+"
+#define FONT_SPACING     "[cmp?*]"
+#define FONT_AVGWIDTH    "[0-9]+"
+#define FONT_REGISTRY    "[^-]+"
+#define FONT_ENCODING    "[^-]+"
+
+#define FONT_REGEXP      ("\\`\\*?[-?*]"     \
+			  FONT_FOUNDRY   "-" \
+			  FONT_FAMILY    "-" \
+			  FONT_WEIGHT_Q  "-" \
+			  FONT_SLANT_Q   "-" \
+			  FONT_SWIDTH    "-" \
+			  FONT_ADSTYLE   "-" \
+			  FONT_PIXELSIZE "-" \
+			  FONT_POINTSIZE "-" \
+			  "[-?*]\\|\\'")
+
+#define FONT_REGEXP_HEAD ("\\`[-?*]"        \
+			  FONT_FOUNDRY  "-" \
+			  FONT_FAMILY   "-" \
+			  FONT_WEIGHT_Q "-" \
+			  FONT_SLANT_Q      \
+			  "\\([-*?]\\|\\'\\)")
+
+#define FONT_REGEXP_SLANT  "-" FONT_SLANT  "-"
+#define FONT_REGEXP_WEIGHT "-" FONT_WEIGHT "-"
+
+LONG 
+x_to_win32_weight (lpw)
+     char * lpw;
+{
+  if (!lpw) return (FW_DONTCARE);
+  
+  if (stricmp (lpw, "bold") == 0)
+    return (FW_BOLD);
+  else if (stricmp (lpw, "demibold") == 0)
+    return (FW_SEMIBOLD);
+  else if (stricmp (lpw, "medium") == 0)
+    return (FW_MEDIUM);
+  else if (stricmp (lpw, "normal") == 0)
+    return (FW_NORMAL);
+  else
+    return (FW_DONTCARE);
+}
+
+char * 
+win32_to_x_weight (fnweight)
+     int fnweight;
+{
+  if (fnweight >= FW_BOLD) 
+    return ("bold");
+  else if (fnweight >= FW_SEMIBOLD) 
+    return ("demibold");
+  else if (fnweight >= FW_MEDIUM) 
+    return ("medium");
+  else  
+    return ("normal");
+}
+
+BOOL 
+win32_to_x_font (lplogfont, lpxstr, len)
+     LOGFONT * lplogfont;
+     char * lpxstr;
+     int len;
+{
+  if (!lpxstr) return (FALSE);
+
+  if (lplogfont) 
+    {
+      int height = (lplogfont->lfHeight * 1440) 
+	/ one_win32_display_info.height_in;
+      int width = (lplogfont->lfWidth * 1440) 
+	/ one_win32_display_info.width_in;
+
+      height = abs (height);
+      _snprintf (lpxstr, len - 1,
+		 "-*-%s-%s-%c-%s-%s-*-%d-*-*-%c-%d-*-*-",
+		 lplogfont->lfFaceName,
+		 win32_to_x_weight (lplogfont->lfWeight),
+		 lplogfont->lfItalic ? 'i' : 'r',
+		 "*", "*", 
+		 height,
+		 ((lplogfont->lfPitchAndFamily & 0x3) == VARIABLE_PITCH) ? 'p' : 'c',
+		 width);
+    } 
+  else 
+    {
+      strncpy (lpxstr, "-*-*-*-*-*-*-*-*-*-*-*-*-*-*-", len - 1);
+    }
+
+  lpxstr[len - 1] = 0;		/* just to be sure */
+  return (TRUE);
+}
+
+BOOL 
+x_to_win32_font (lpxstr, lplogfont)
+     char * lpxstr;
+     LOGFONT * lplogfont;
+{
+  if (!lplogfont) return (FALSE);
+  
+  memset (lplogfont, 0, sizeof (*lplogfont));
+  
+  lplogfont->lfCharSet = OEM_CHARSET;
+  lplogfont->lfOutPrecision = OUT_DEFAULT_PRECIS;
+  lplogfont->lfClipPrecision = CLIP_DEFAULT_PRECIS;
+  lplogfont->lfQuality = DEFAULT_QUALITY;
+  
+  if (lpxstr && *lpxstr == '-') lpxstr++;
+  
+  {
+    int fields;
+    char name[50], weight[20], slant, pitch, height[10], width[10];
+    
+    fields = (lpxstr
+	      ? sscanf (lpxstr, 
+			"%*[^-]-%[^-]-%[^-]-%c-%*[^-]-%*[^-]-%*[^-]-%[^-]-%*[^-]-%*[^-]-%c-%[^-]",
+			name, weight, &slant, height, &pitch, width)
+	      : 0);
+    
+    if (fields == EOF) return (FALSE);
+    
+    if (fields > 0 && name[0] != '*') 
+      {
+	strncpy (lplogfont->lfFaceName, name, LF_FACESIZE);
+      } 
+    else 
+      {
+	lplogfont->lfFaceName[0] = 0;
+      }
+    
+    fields--;
+    
+    lplogfont->lfWeight = x_to_win32_weight((fields > 0 ? weight : ""));
+    
+    fields--;
+    
+    lplogfont->lfItalic = (fields > 0 && slant == 'i');
+    
+    fields--;
+    
+    if (fields > 0 && height[0] != '*')
+      lplogfont->lfHeight = (atoi (height) * one_win32_display_info.height_in) / 1440;
+    
+    fields--;
+    
+    lplogfont->lfPitchAndFamily = (fields > 0 && pitch == 'p') ? VARIABLE_PITCH : FIXED_PITCH;
+    
+    fields--;
+    
+    if (fields > 0 && width[0] != '*')
+      lplogfont->lfWidth = (atoi (width) * one_win32_display_info.width_in) / 1440;
+  }
+  
+  return (TRUE);
+}
+
+BOOL 
+win32_font_match (lpszfont1, lpszfont2)
+    char * lpszfont1;
+    char * lpszfont2;
+{
+  char * s1 = lpszfont1, *e1;
+  char * s2 = lpszfont2, *e2;
+  
+  if (s1 == NULL || s2 == NULL) return (FALSE);
+  
+  if (*s1 == '-') s1++;
+  if (*s2 == '-') s2++;
+  
+  while (1) 
+    {
+      int len1, len2;
+
+      e1 = strchr (s1, '-');
+      e2 = strchr (s2, '-');
+
+      if (e1 == NULL || e2 == NULL) return (TRUE);
+
+      len1 = e1 - s1;
+      len2 = e2 - s2;
+
+      if (*s1 != '*' && *s2 != '*'
+	  && (len1 != len2 || strnicmp (s1, s2, len1) != 0))
+	return (FALSE);
+
+      s1 = e1 + 1;
+      s2 = e2 + 1;
+    }
+}
+
+typedef struct enumfont_t 
+{
+  HDC hdc;
+  int numFonts;
+  XFontStruct *size_ref;
+  Lisp_Object *pattern;
+  Lisp_Object *head;
+  Lisp_Object *tail;
+} enumfont_t;
+
+int CALLBACK 
+enum_font_cb2 (lplf, lptm, FontType, lpef)
+    ENUMLOGFONT * lplf;
+    NEWTEXTMETRIC * lptm;
+    int FontType;
+    enumfont_t * lpef;
+{
+  if (lplf->elfLogFont.lfStrikeOut || lplf->elfLogFont.lfUnderline
+      || (lplf->elfLogFont.lfCharSet != ANSI_CHARSET && lplf->elfLogFont.lfCharSet != OEM_CHARSET))
+    return (1);
+  
+  /*    if (!lpef->size_ref || lptm->tmMaxCharWidth == FONT_WIDTH (lpef->size_ref)) */
+  {
+    char buf[100];
+
+    if (!win32_to_x_font (lplf, buf, 100)) return (0);
+
+    if (NILP (*(lpef->pattern)) || win32_font_match (buf, XSTRING (*(lpef->pattern))->data))
+      {
+	*lpef->tail = Fcons (build_string (buf), Qnil);
+	lpef->tail = &XCONS (*lpef->tail)->cdr;
+	lpef->numFonts++;
+      }
+  }
+  
+  return (1);
+}
+
+int CALLBACK 
+enum_font_cb1 (lplf, lptm, FontType, lpef)
+     ENUMLOGFONT * lplf;
+     NEWTEXTMETRIC * lptm;
+     int FontType;
+     enumfont_t * lpef;
+{
+  return EnumFontFamilies (lpef->hdc,
+			   lplf->elfLogFont.lfFaceName,
+			   (FONTENUMPROC) enum_font_cb2,
+			   (LPARAM) lpef);
+}
+
+
+DEFUN ("x-list-fonts", Fx_list_fonts, Sx_list_fonts, 1, 3, 0,
+  "Return a list of the names of available fonts matching PATTERN.\n\
+If optional arguments FACE and FRAME are specified, return only fonts\n\
+the same size as FACE on FRAME.\n\
+\n\
+PATTERN is a string, perhaps with wildcard characters;\n\
+  the * character matches any substring, and\n\
+  the ? character matches any single character.\n\
+  PATTERN is case-insensitive.\n\
+FACE is a face name--a symbol.\n\
+\n\
+The return value is a list of strings, suitable as arguments to\n\
+set-face-font.\n\
+\n\
+Fonts Emacs can't use (i.e. proportional fonts) may or may not be excluded\n\
+even if they match PATTERN and FACE.")
+  (pattern, face, frame)
+    Lisp_Object pattern, face, frame;
+{
+  int num_fonts;
+  char **names;
+  XFontStruct *info;
+  XFontStruct *size_ref;
+  Lisp_Object namelist;
+  Lisp_Object list;
+  FRAME_PTR f;
+  enumfont_t ef;
+
+  CHECK_STRING (pattern, 0);
+  if (!NILP (face))
+    CHECK_SYMBOL (face, 1);
+
+  f = check_x_frame (frame);
+
+  /* Determine the width standard for comparison with the fonts we find.  */
+
+  if (NILP (face))
+    size_ref = 0;
+  else
+    {
+      int face_id;
+
+      /* Don't die if we get called with a terminal frame.  */
+      if (! FRAME_WIN32_P (f))
+	error ("non-win32 frame used in `x-list-fonts'");
+
+      face_id = face_name_id_number (f, face);
+
+      if (face_id < 0 || face_id >= FRAME_N_PARAM_FACES (f)
+	  || FRAME_PARAM_FACES (f) [face_id] == 0)
+	size_ref = f->output_data.win32->font;
+      else
+	{
+	  size_ref = FRAME_PARAM_FACES (f) [face_id]->font;
+	  if (size_ref == (XFontStruct *) (~0))
+	    size_ref = f->output_data.win32->font;
+	}
+    }
+
+  /* See if we cached the result for this particular query.  */
+  list = Fassoc (pattern,
+		 XCONS (FRAME_WIN32_DISPLAY_INFO (f)->name_list_element)->cdr);
+
+  /* We have info in the cache for this PATTERN.  */
+  if (!NILP (list))
+    {
+      Lisp_Object tem, newlist;
+
+      /* We have info about this pattern.  */
+      list = XCONS (list)->cdr;
+
+      if (size_ref == 0)
+	return list;
+
+      BLOCK_INPUT;
+
+      /* Filter the cached info and return just the fonts that match FACE.  */
+      newlist = Qnil;
+      for (tem = list; CONSP (tem); tem = XCONS (tem)->cdr)
+	{
+	  XFontStruct *thisinfo;
+
+          thisinfo = win32_load_font (FRAME_WIN32_DISPLAY_INFO (f), XSTRING (XCONS (tem)->car)->data);
+
+          if (thisinfo && same_size_fonts (thisinfo, size_ref))
+	    newlist = Fcons (XCONS (tem)->car, newlist);
+
+	  win32_unload_font (FRAME_WIN32_DISPLAY_INFO (f), thisinfo);
+        }
+
+      UNBLOCK_INPUT;
+
+      return newlist;
+    }
+
+  BLOCK_INPUT;
+
+  namelist = Qnil;
+  ef.pattern = &pattern;
+  ef.tail = ef.head = &namelist;
+  ef.numFonts = 0;
+
+  {
+    ef.hdc = my_get_dc (FRAME_WIN32_WINDOW (f));
+
+    EnumFontFamilies (ef.hdc, NULL, (FONTENUMPROC) enum_font_cb1, (LPARAM)&ef);
+    
+    ReleaseDC  (FRAME_WIN32_WINDOW (f), ef.hdc);
+  }
+
+  UNBLOCK_INPUT;
+
+  if (ef.numFonts)
+    {
+      int i;
+      Lisp_Object cur;
+
+      /* Make a list of all the fonts we got back.
+	 Store that in the font cache for the display.  */
+      XCONS (FRAME_WIN32_DISPLAY_INFO (f)->name_list_element)->cdr
+	= Fcons (Fcons (pattern, namelist),
+		 XCONS (FRAME_WIN32_DISPLAY_INFO (f)->name_list_element)->cdr);
+
+      /* Make a list of the fonts that have the right width.  */
+      list = Qnil;
+      cur=namelist;
+      for (i = 0; i < ef.numFonts; i++)
+        {
+	  int keeper;
+
+	  if (!size_ref)
+	    keeper = 1;
+	  else
+	    {
+	      XFontStruct *thisinfo;
+
+	      BLOCK_INPUT;
+	      thisinfo = win32_load_font (FRAME_WIN32_DISPLAY_INFO (f), XSTRING (Fcar (cur))->data);
+
+	      keeper = thisinfo && same_size_fonts (thisinfo, size_ref);
+
+	      win32_unload_font (FRAME_WIN32_DISPLAY_INFO (f), thisinfo);
+
+	      UNBLOCK_INPUT;
+	    }
+          if (keeper)
+	    list = Fcons (build_string (XSTRING (Fcar (cur))->data), list);
+
+	  cur = Fcdr (cur);
+        }
+      list = Fnreverse (list);
+    }
+
+  return list;
+}
+
+DEFUN ("x-color-defined-p", Fx_color_defined_p, Sx_color_defined_p, 1, 2, 0,
+       "Return non-nil if color COLOR is supported on frame FRAME.\n\
+If FRAME is omitted or nil, use the selected frame.")
+  (color, frame)
+     Lisp_Object color, frame;
+{
+  COLORREF foo;
+  FRAME_PTR f = check_x_frame (frame);
+
+  CHECK_STRING (color, 1);
+
+  if (defined_color (f, XSTRING (color)->data, &foo, 0))
+    return Qt;
+  else
+    return Qnil;
+}
+
+DEFUN ("x-color-values", Fx_color_values, Sx_color_values, 1, 2, 0,
+  "Return a description of the color named COLOR on frame FRAME.\n\
+The value is a list of integer RGB values--(RED GREEN BLUE).\n\
+These values appear to range from 0 to 65280 or 65535, depending\n\
+on the system; white is (65280 65280 65280) or (65535 65535 65535).\n\
+If FRAME is omitted or nil, use the selected frame.")
+  (color, frame)
+     Lisp_Object color, frame;
+{
+  COLORREF foo;
+  FRAME_PTR f = check_x_frame (frame);
+
+  CHECK_STRING (color, 1);
+
+  if (defined_color (f, XSTRING (color)->data, &foo, 0))
+    {
+      Lisp_Object rgb[3];
+
+      rgb[0] = make_number (GetRValue (foo));
+      rgb[1] = make_number (GetGValue (foo));
+      rgb[2] = make_number (GetBValue (foo));
+      return Flist (3, rgb);
+    }
+  else
+    return Qnil;
+}
+
+DEFUN ("x-display-color-p", Fx_display_color_p, Sx_display_color_p, 0, 1, 0,
+  "Return t if the X display supports color.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  if ((dpyinfo->n_planes * dpyinfo->n_cbits) <= 2)
+    return Qnil;
+
+  return Qt;
+}
+
+DEFUN ("x-display-grayscale-p", Fx_display_grayscale_p, Sx_display_grayscale_p,
+  0, 1, 0,
+  "Return t if the X display supports shades of gray.\n\
+Note that color displays do support shades of gray.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  if ((dpyinfo->n_planes * dpyinfo->n_cbits) <= 1)
+    return Qnil;
+
+  return Qt;
+}
+
+DEFUN ("x-display-pixel-width", Fx_display_pixel_width, Sx_display_pixel_width,
+  0, 1, 0,
+  "Returns the width in pixels of the X display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return make_number (dpyinfo->width);
+}
+
+DEFUN ("x-display-pixel-height", Fx_display_pixel_height,
+  Sx_display_pixel_height, 0, 1, 0,
+  "Returns the height in pixels of the X display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return make_number (dpyinfo->height);
+}
+
+DEFUN ("x-display-planes", Fx_display_planes, Sx_display_planes,
+  0, 1, 0,
+  "Returns the number of bitplanes of the display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return make_number (dpyinfo->n_planes * dpyinfo->n_cbits);
+}
+
+DEFUN ("x-display-color-cells", Fx_display_color_cells, Sx_display_color_cells,
+  0, 1, 0,
+  "Returns the number of color cells of the display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+  HDC hdc;
+  int cap;
+
+  hdc = my_get_dc (dpyinfo->root_window);
+  
+  cap = GetDeviceCaps (hdc,NUMCOLORS);
+  
+  ReleaseDC (dpyinfo->root_window, hdc);
+  
+  return make_number (cap);
+}
+
+DEFUN ("x-server-max-request-size", Fx_server_max_request_size,
+       Sx_server_max_request_size,
+  0, 1, 0,
+  "Returns the maximum request size of the server of display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return make_number (1);
+}
+
+DEFUN ("x-server-vendor", Fx_server_vendor, Sx_server_vendor, 0, 1, 0,
+  "Returns the vendor ID string of the Win32 system (Microsoft).\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+  char *vendor = "Microsoft Corp.";
+
+  if (! vendor) vendor = "";
+  return build_string (vendor);
+}
+
+DEFUN ("x-server-version", Fx_server_version, Sx_server_version, 0, 1, 0,
+  "Returns the version numbers of the server of display DISPLAY.\n\
+The value is a list of three integers: the major and minor\n\
+version numbers, and the vendor-specific release\n\
+number.  See also the function `x-server-vendor'.\n\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return Fcons (make_number (nt_major_version),
+		Fcons (make_number (nt_minor_version), Qnil));
+}
+
+DEFUN ("x-display-screens", Fx_display_screens, Sx_display_screens, 0, 1, 0,
+  "Returns the number of screens on the server of display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return make_number (1);
+}
+
+DEFUN ("x-display-mm-height", Fx_display_mm_height, Sx_display_mm_height, 0, 1, 0,
+  "Returns the height in millimeters of the X display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+  HDC hdc;
+  int cap;
+
+  hdc = my_get_dc (dpyinfo->root_window);
+  
+  cap = GetDeviceCaps (hdc, VERTSIZE);
+  
+  ReleaseDC (dpyinfo->root_window, hdc);
+  
+  return make_number (cap);
+}
+
+DEFUN ("x-display-mm-width", Fx_display_mm_width, Sx_display_mm_width, 0, 1, 0,
+  "Returns the width in millimeters of the X display DISPLAY.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  HDC hdc;
+  int cap;
+
+  hdc = my_get_dc (dpyinfo->root_window);
+  
+  cap = GetDeviceCaps (hdc, HORZSIZE);
+  
+  ReleaseDC (dpyinfo->root_window, hdc);
+  
+  return make_number (cap);
+}
+
+DEFUN ("x-display-backing-store", Fx_display_backing_store,
+  Sx_display_backing_store, 0, 1, 0,
+  "Returns an indication of whether display DISPLAY does backing store.\n\
+The value may be `always', `when-mapped', or `not-useful'.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  return intern ("not-useful");
+}
+
+DEFUN ("x-display-visual-class", Fx_display_visual_class,
+  Sx_display_visual_class, 0, 1, 0,
+  "Returns the visual class of the display DISPLAY.\n\
+The value is one of the symbols `static-gray', `gray-scale',\n\
+`static-color', `pseudo-color', `true-color', or `direct-color'.\n\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+	(display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+#if 0
+  switch (dpyinfo->visual->class)
+    {
+    case StaticGray:  return (intern ("static-gray"));
+    case GrayScale:   return (intern ("gray-scale"));
+    case StaticColor: return (intern ("static-color"));
+    case PseudoColor: return (intern ("pseudo-color"));
+    case TrueColor:   return (intern ("true-color"));
+    case DirectColor: return (intern ("direct-color"));
+    default:
+      error ("Display has an unknown visual class");
+    }
+#endif
+
+  error ("Display has an unknown visual class");
+}
+
+DEFUN ("x-display-save-under", Fx_display_save_under,
+  Sx_display_save_under, 0, 1, 0,
+  "Returns t if the display DISPLAY supports the save-under feature.\n\
+The optional argument DISPLAY specifies which display to ask about.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If omitted or nil, that stands for the selected frame's display.")
+  (display)
+     Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return Qnil;
+}
+
+int
+x_pixel_width (f)
+     register struct frame *f;
+{
+  return PIXEL_WIDTH (f);
+}
+
+int
+x_pixel_height (f)
+     register struct frame *f;
+{
+  return PIXEL_HEIGHT (f);
+}
+
+int
+x_char_width (f)
+     register struct frame *f;
+{
+  return FONT_WIDTH (f->output_data.win32->font);
+}
+
+int
+x_char_height (f)
+     register struct frame *f;
+{
+  return f->output_data.win32->line_height;
+}
+
+int
+x_screen_planes (frame)
+     Lisp_Object frame;
+{
+  return (FRAME_WIN32_DISPLAY_INFO (XFRAME (frame))->n_planes * 
+	  FRAME_WIN32_DISPLAY_INFO (XFRAME (frame))->n_cbits);
+}
+
+/* Return the display structure for the display named NAME.
+   Open a new connection if necessary.  */
+
+struct win32_display_info *
+x_display_info_for_name (name)
+     Lisp_Object name;
+{
+  Lisp_Object names;
+  struct win32_display_info *dpyinfo;
+
+  CHECK_STRING (name, 0);
+
+  for (dpyinfo = &one_win32_display_info, names = win32_display_name_list;
+       dpyinfo;
+       dpyinfo = dpyinfo->next, names = XCONS (names)->cdr)
+    {
+      Lisp_Object tem;
+      tem = Fstring_equal (XCONS (XCONS (names)->car)->car, name);
+      if (!NILP (tem))
+	return dpyinfo;
+    }
+
+  /* Use this general default value to start with.  */
+  Vx_resource_name = Vinvocation_name;
+
+  validate_x_resource_name ();
+
+  dpyinfo = win32_term_init (name, (unsigned char *)0,
+			     (char *) XSTRING (Vx_resource_name)->data);
+
+  if (dpyinfo == 0)
+    error ("Cannot connect to server %s", XSTRING (name)->data);
+
+  XSETFASTINT (Vwindow_system_version, 3);
+
+  return dpyinfo;
+}
+
+DEFUN ("x-open-connection", Fx_open_connection, Sx_open_connection,
+       1, 3, 0, "Open a connection to a server.\n\
+DISPLAY is the name of the display to connect to.\n\
+Optional second arg XRM-STRING is a string of resources in xrdb format.\n\
+If the optional third arg MUST-SUCCEED is non-nil,\n\
+terminate Emacs if we can't open the connection.")
+  (display, xrm_string, must_succeed)
+     Lisp_Object display, xrm_string, must_succeed;
+{
+  unsigned int n_planes;
+  unsigned char *xrm_option;
+  struct win32_display_info *dpyinfo;
+
+  CHECK_STRING (display, 0);
+  if (! NILP (xrm_string))
+    CHECK_STRING (xrm_string, 1);
+
+  Vwin32_color_map = Fwin32_default_color_map ();
+
+  if (! NILP (xrm_string))
+    xrm_option = (unsigned char *) XSTRING (xrm_string)->data;
+  else
+    xrm_option = (unsigned char *) 0;
+
+  /* Use this general default value to start with.  */
+  Vx_resource_name = Vinvocation_name;
+
+  validate_x_resource_name ();
+
+  /* This is what opens the connection and sets x_current_display.
+     This also initializes many symbols, such as those used for input.  */
+  dpyinfo = win32_term_init (display, xrm_option,
+			     (char *) XSTRING (Vx_resource_name)->data);
+
+  if (dpyinfo == 0)
+    {
+      if (!NILP (must_succeed))
+	fatal ("Cannot connect to server %s.\n",
+	       XSTRING (display)->data);
+      else
+	error ("Cannot connect to server %s", XSTRING (display)->data);
+    }
+
+  XSETFASTINT (Vwindow_system_version, 3);
+  return Qnil;
+}
+
+DEFUN ("x-close-connection", Fx_close_connection,
+       Sx_close_connection, 1, 1, 0,
+   "Close the connection to DISPLAY's server.\n\
+For DISPLAY, specify either a frame or a display name (a string).\n\
+If DISPLAY is nil, that stands for the selected frame's display.")
+  (display)
+  Lisp_Object display;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+  struct win32_display_info *tail;
+  int i;
+
+  if (dpyinfo->reference_count > 0)
+    error ("Display still has frames on it");
+
+  BLOCK_INPUT;
+  /* Free the fonts in the font table.  */
+  for (i = 0; i < dpyinfo->n_fonts; i++)
+    {
+      if (dpyinfo->font_table[i].name)
+	free (dpyinfo->font_table[i].name);
+      /* Don't free the full_name string;
+	 it is always shared with something else.  */
+      win32_unload_font (dpyinfo, dpyinfo->font_table[i].font);
+    }
+  x_destroy_all_bitmaps (dpyinfo);
+
+  x_delete_display (dpyinfo);
+  UNBLOCK_INPUT;
+
+  return Qnil;
+}
+
+DEFUN ("x-display-list", Fx_display_list, Sx_display_list, 0, 0, 0,
+  "Return the list of display names that Emacs has connections to.")
+  ()
+{
+  Lisp_Object tail, result;
+
+  result = Qnil;
+  for (tail = win32_display_name_list; ! NILP (tail); tail = XCONS (tail)->cdr)
+    result = Fcons (XCONS (XCONS (tail)->car)->car, result);
+
+  return result;
+}
+
+DEFUN ("x-synchronize", Fx_synchronize, Sx_synchronize, 1, 2, 0,
+   "If ON is non-nil, report errors as soon as the erring request is made.\n\
+If ON is nil, allow buffering of requests.\n\
+This is a noop on Win32 systems.\n\
+The optional second argument DISPLAY specifies which display to act on.\n\
+DISPLAY should be either a frame or a display name (a string).\n\
+If DISPLAY is omitted or nil, that stands for the selected frame's display.")
+  (on, display)
+    Lisp_Object display, on;
+{
+  struct win32_display_info *dpyinfo = check_x_display_info (display);
+
+  return Qnil;
+}
+
+
+/* These are the win32 specialized functions */
+
+DEFUN ("win32-select-font", Fwin32_select_font, Swin32_select_font, 0, 1, 0,
+   "This will display the Win32 font dialog and return an X font string corresponding to the selection.")
+  (frame)
+     Lisp_Object frame;
+{
+  FRAME_PTR f = check_x_frame (frame);
+  CHOOSEFONT cf;
+  LOGFONT lf;
+  char buf[100];
+
+  bzero (&cf, sizeof (cf));
+
+  cf.lStructSize = sizeof (cf);
+  cf.hwndOwner = FRAME_WIN32_WINDOW (f);
+  cf.Flags = CF_FIXEDPITCHONLY | CF_FORCEFONTEXIST | CF_SCREENFONTS;
+  cf.lpLogFont = &lf;
+
+  if (!ChooseFont (&cf) || !win32_to_x_font (&lf, buf, 100))
+      return Qnil;
+
+  return build_string (buf);
+}
+
+
+syms_of_win32fns ()
+{
+  /* The section below is built by the lisp expression at the top of the file,
+     just above where these variables are declared.  */
+  /*&&& init symbols here &&&*/
+  Qauto_raise = intern ("auto-raise");
+  staticpro (&Qauto_raise);
+  Qauto_lower = intern ("auto-lower");
+  staticpro (&Qauto_lower);
+  Qbackground_color = intern ("background-color");
+  staticpro (&Qbackground_color);
+  Qbar = intern ("bar");
+  staticpro (&Qbar);
+  Qborder_color = intern ("border-color");
+  staticpro (&Qborder_color);
+  Qborder_width = intern ("border-width");
+  staticpro (&Qborder_width);
+  Qbox = intern ("box");
+  staticpro (&Qbox);
+  Qcursor_color = intern ("cursor-color");
+  staticpro (&Qcursor_color);
+  Qcursor_type = intern ("cursor-type");
+  staticpro (&Qcursor_type);
+  Qfont = intern ("font");
+  staticpro (&Qfont);
+  Qforeground_color = intern ("foreground-color");
+  staticpro (&Qforeground_color);
+  Qgeometry = intern ("geometry");
+  staticpro (&Qgeometry);
+  Qicon_left = intern ("icon-left");
+  staticpro (&Qicon_left);
+  Qicon_top = intern ("icon-top");
+  staticpro (&Qicon_top);
+  Qicon_type = intern ("icon-type");
+  staticpro (&Qicon_type);
+  Qicon_name = intern ("icon-name");
+  staticpro (&Qicon_name);
+  Qinternal_border_width = intern ("internal-border-width");
+  staticpro (&Qinternal_border_width);
+  Qleft = intern ("left");
+  staticpro (&Qleft);
+  Qmouse_color = intern ("mouse-color");
+  staticpro (&Qmouse_color);
+  Qnone = intern ("none");
+  staticpro (&Qnone);
+  Qparent_id = intern ("parent-id");
+  staticpro (&Qparent_id);
+  Qscroll_bar_width = intern ("scroll-bar-width");
+  staticpro (&Qscroll_bar_width);
+  Qsuppress_icon = intern ("suppress-icon");
+  staticpro (&Qsuppress_icon);
+  Qtop = intern ("top");
+  staticpro (&Qtop);
+  Qundefined_color = intern ("undefined-color");
+  staticpro (&Qundefined_color);
+  Qvertical_scroll_bars = intern ("vertical-scroll-bars");
+  staticpro (&Qvertical_scroll_bars);
+  Qvisibility = intern ("visibility");
+  staticpro (&Qvisibility);
+  Qwindow_id = intern ("window-id");
+  staticpro (&Qwindow_id);
+  Qx_frame_parameter = intern ("x-frame-parameter");
+  staticpro (&Qx_frame_parameter);
+  Qx_resource_name = intern ("x-resource-name");
+  staticpro (&Qx_resource_name);
+  Quser_position = intern ("user-position");
+  staticpro (&Quser_position);
+  Quser_size = intern ("user-size");
+  staticpro (&Quser_size);
+  Qdisplay = intern ("display");
+  staticpro (&Qdisplay);
+  /* This is the end of symbol initialization.  */
+
+  Fput (Qundefined_color, Qerror_conditions,
+	Fcons (Qundefined_color, Fcons (Qerror, Qnil)));
+  Fput (Qundefined_color, Qerror_message,
+	build_string ("Undefined color"));
+
+  DEFVAR_LISP ("win32-color-map", &Vwin32_color_map,
+	       "A array of color name mappings for windows.");
+  Vwin32_color_map = Qnil;
+
+  init_x_parm_symbols ();
+
+  DEFVAR_LISP ("x-bitmap-file-path", &Vx_bitmap_file_path,
+    "List of directories to search for bitmap files for win32.");
+  Vx_bitmap_file_path = decode_env_path ((char *) 0, "PATH");
+
+  DEFVAR_LISP ("x-pointer-shape", &Vx_pointer_shape,
+    "The shape of the pointer when over text.\n\
+Changing the value does not affect existing frames\n\
+unless you set the mouse color.");
+  Vx_pointer_shape = Qnil;
+
+  DEFVAR_LISP ("x-resource-name", &Vx_resource_name,
+    "The name Emacs uses to look up resources; for internal use only.\n\
+`x-get-resource' uses this as the first component of the instance name\n\
+when requesting resource values.\n\
+Emacs initially sets `x-resource-name' to the name under which Emacs\n\
+was invoked, or to the value specified with the `-name' or `-rn'\n\
+switches, if present.");
+  Vx_resource_name = Qnil;
+
+  Vx_nontext_pointer_shape = Qnil;
+
+  Vx_mode_pointer_shape = Qnil;
+
+  DEFVAR_INT ("x-sensitive-text-pointer-shape",
+	      &Vx_sensitive_text_pointer_shape,
+	      "The shape of the pointer when over mouse-sensitive text.\n\
+This variable takes effect when you create a new frame\n\
+or when you set the mouse color.");
+  Vx_sensitive_text_pointer_shape = Qnil;
+
+  DEFVAR_LISP ("x-cursor-fore-pixel", &Vx_cursor_fore_pixel,
+	       "A string indicating the foreground color of the cursor box.");
+  Vx_cursor_fore_pixel = Qnil;
+
+  DEFVAR_LISP ("x-no-window-manager", &Vx_no_window_manager,
+	       "Non-nil if no window manager is in use.\n\
+Emacs doesn't try to figure this out; this is always nil\n\
+unless you set it to something else.");
+  /* We don't have any way to find this out, so set it to nil
+     and maybe the user would like to set it to t.  */
+  Vx_no_window_manager = Qnil;
+
+  defsubr (&Sx_get_resource);
+  defsubr (&Sx_list_fonts);
+  defsubr (&Sx_display_color_p);
+  defsubr (&Sx_display_grayscale_p);
+  defsubr (&Sx_color_defined_p);
+  defsubr (&Sx_color_values);
+  defsubr (&Sx_server_max_request_size);
+  defsubr (&Sx_server_vendor);
+  defsubr (&Sx_server_version);
+  defsubr (&Sx_display_pixel_width);
+  defsubr (&Sx_display_pixel_height);
+  defsubr (&Sx_display_mm_width);
+  defsubr (&Sx_display_mm_height);
+  defsubr (&Sx_display_screens);
+  defsubr (&Sx_display_planes);
+  defsubr (&Sx_display_color_cells);
+  defsubr (&Sx_display_visual_class);
+  defsubr (&Sx_display_backing_store);
+  defsubr (&Sx_display_save_under);
+  defsubr (&Sx_parse_geometry);
+  defsubr (&Sx_create_frame);
+  defsubr (&Sfocus_frame);
+  defsubr (&Sunfocus_frame);
+  defsubr (&Sx_open_connection);
+  defsubr (&Sx_close_connection);
+  defsubr (&Sx_display_list);
+  defsubr (&Sx_synchronize);
+
+  /* Win32 specific functions */
+
+  defsubr (&Swin32_select_font);
+}
+
+#undef abort
+
+void 
+win32_abort()
+{
+    MessageBox (NULL,
+		"A fatal error has occurred - aborting!",
+		"Emacs Abort Dialog",
+		MB_OK|MB_ICONEXCLAMATION);
+    abort();
+}