view src/toolbar-msw.c @ 283:fa3d41851a08

Added tag r21-0b39 for changeset c42ec1d1cded
author cvs
date Mon, 13 Aug 2007 10:33:19 +0200
parents 7df0dd720c89
children 558f606b08ae
line wrap: on
line source

/* toolbar implementation -- mswindows interface.
   Copyright (C) 1995 Board of Trustees, University of Illinois.
   Copyright (C) 1995 Sun Microsystems, Inc.
   Copyright (C) 1995, 1996 Ben Wing.
   Copyright (C) 1996 Chuck Thompson.
   Copyright (C) 1998 Andy Piper.

This file is part of XEmacs.

XEmacs is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation; either version 2, or (at your option) any
later version.

XEmacs is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
for more details.

You should have received a copy of the GNU General Public License
along with XEmacs; see the file COPYING.  If not, write to
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Boston, MA 02111-1307, USA.  */

/* This implementation by Andy Piper <andyp@parallax.co.uk>, with bits
   borrowed from toolbar-x.c */

/* Synched up with: Not in FSF. */

#include <config.h>
#include "lisp.h"

#include "faces.h"
#include "frame.h"
#include "toolbar.h"
#include "window.h"
#include "gui.h"
#include "elhash.h"
#include "console-msw.h"
#include "glyphs-msw.h"
#include "objects-msw.h"

/* Why did Kirill choose this range ? */
#define TOOLBAR_ITEM_ID_MIN 0x4000
#define TOOLBAR_ITEM_ID_MAX 0x7FFF
#define TOOLBAR_ITEM_ID_BITS(x) (((x) & 0x3FFF) | 0x4000)
#define TOOLBAR_ID_BIAS 16
#define TOOLBAR_HANDLE(f,p) \
GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS + p)
#ifndef TB_SETIMAGELIST
#define TB_SETIMAGELIST (WM_USER + 48)
#define TB_GETIMAGELIST (WM_USER + 49)
#define TB_SETPADDING   (WM_USER + 87)
#endif

#define SET_TOOLBAR_WAS_VISIBLE_FLAG(frame, pos, flag)			\
  do {									\
    switch (pos)							\
      {									\
      case TOP_TOOLBAR:							\
	(frame)->top_toolbar_was_visible = flag;			\
	break;								\
      case BOTTOM_TOOLBAR:						\
	(frame)->bottom_toolbar_was_visible = flag;			\
	break;								\
      case LEFT_TOOLBAR:						\
	(frame)->left_toolbar_was_visible = flag;			\
	break;								\
      case RIGHT_TOOLBAR:						\
	(frame)->right_toolbar_was_visible = flag;			\
	break;								\
      default:								\
	abort ();							\
      }									\
  } while (0)

static int
allocate_toolbar_item_id (struct frame* f, struct toolbar_button* button,
			  enum toolbar_pos pos)
{
  /* hmm what do we generate an id based on */
  int id = TOOLBAR_ITEM_ID_BITS (internal_hash (button->callback, 0));
  while (!NILP (Fgethash (make_int (id),
			  FRAME_MSWINDOWS_TOOLBAR_HASHTABLE (f), Qnil)))
    {
      id = TOOLBAR_ITEM_ID_BITS (id + 1);
    }
  return id;
}

static void
mswindows_clear_toolbar (struct frame *f, enum toolbar_pos pos,
			 int thickness_change)
{
  HIMAGELIST ilist=NULL;
  int i;
  HWND toolbarwnd = TOOLBAR_HANDLE(f, pos);
  if (toolbarwnd)
    {
      TBBUTTON info;
      
      /* delete the buttons and remove the command from the hashtable*/
      i = SendMessage (toolbarwnd, TB_BUTTONCOUNT, 0, 0);
      for (i--; i >= 0; i--)
	{
	  SendMessage (toolbarwnd, TB_GETBUTTON, (WPARAM)i, 
		       (LPARAM)&info);
	  Fremhash(make_int(info.idCommand), 
		   FRAME_MSWINDOWS_TOOLBAR_HASHTABLE(f));
	  SendMessage (toolbarwnd, TB_DELETEBUTTON, (WPARAM)i, 0);
	}
	  
      /* finally get rid of the image list assuming it clears up its
         bitmaps */
      SendMessage (toolbarwnd, TB_GETIMAGELIST, 0, (LONG) &ilist);
      if (ilist)
	{
	  ImageList_Destroy(ilist);
	}
      SendMessage (toolbarwnd, TB_SETIMAGELIST, 0, (LPARAM)NULL);

      ShowWindow(toolbarwnd, SW_HIDE);
    }

  FRAME_MSWINDOWS_TOOLBAR_CHECKSUM(f,pos)=0;
  SET_TOOLBAR_WAS_VISIBLE_FLAG (f, pos, 0);
}

static void
mswindows_output_toolbar (struct frame *f, enum toolbar_pos pos)
{
  int x, y, bar_width, bar_height, vert;
  int width=-1, height=-1, bmwidth=-1, bmheight=-1;
  int border_width = FRAME_REAL_TOOLBAR_BORDER_WIDTH (f, pos);
  Lisp_Object button, window, glyph, instance;
  int nbuttons=0;
  int shadow_thickness = 2;	/* get this from somewhere else? */
  int window_frame_width = 3;
  unsigned int checksum=0;
  struct window *w;
  TBBUTTON* button_tbl, *tbbutton;
  HIMAGELIST ilist=NULL;
  HWND toolbarwnd=NULL;

  get_toolbar_coords (f, pos, &x, &y, &bar_width, &bar_height, &vert, 0);

  if (x==1)
    x=0;

  window = FRAME_LAST_NONMINIBUF_WINDOW (f);
  w = XWINDOW (window);

  toolbarwnd = TOOLBAR_HANDLE(f,pos);
  
  /* set button sizes based on bar size */
  if (vert)
    {
      width = height = bar_width;
      bmwidth = bmheight = width - (border_width + shadow_thickness) * 2;
    }
  else
    {
      height = width = bar_height - window_frame_width * 2; 
      bmwidth = bmheight = width - (border_width + shadow_thickness) * 2; 
    }

  button = FRAME_TOOLBAR_DATA (f, pos)->toolbar_buttons;

  /* First loop over all of the buttons to determine how many there
     are. This loop will also make sure that all instances are
     instantiated so when we actually output them they will come up
     immediately. */
  while (!NILP (button))
    {
      struct toolbar_button *tb = XTOOLBAR_BUTTON (button);
      checksum = HASH3 (checksum, 
			internal_hash (get_toolbar_button_glyph(w, tb), 0),
			internal_hash (tb->callback, 0));
      button = tb->next;
      nbuttons++;
    }

  /* only rebuild if something has changed */
  if (!toolbarwnd || FRAME_MSWINDOWS_TOOLBAR_CHECKSUM(f,pos)!=checksum)
    {
      /* remove the old one */
      mswindows_clear_toolbar (f, pos, 0);

      FRAME_MSWINDOWS_TOOLBAR_CHECKSUM(f,pos)=checksum;

      /* build up the data required by win32 fns. */
      button_tbl = xnew_array_and_zero (TBBUTTON, nbuttons);
      button = FRAME_TOOLBAR_DATA (f, pos)->toolbar_buttons;
      tbbutton = button_tbl;

      while (!NILP (button))
	{
	  struct toolbar_button *tb = XTOOLBAR_BUTTON (button);
	  
	  tbbutton->idCommand = allocate_toolbar_item_id (f, tb, pos);
	  tbbutton->fsState=tb->enabled ? TBSTATE_ENABLED 
	    : TBSTATE_INDETERMINATE;
	  tbbutton->fsStyle=tb->blank ? TBSTYLE_SEP : TBSTYLE_BUTTON;
	  tbbutton->dwData=0; 
	  tbbutton->iString=0;
	  
	  /* note that I am not doing the button size here. This is
             because it is slightly out of my control and the main
             place they are used is in redisplay for getting events
             over toolbar buttons. Since the right way to do help echo
             is with tooltips I'm not going to bother with the extra
             work involved. */
	  
	  /* mess with the button image */
	  glyph = get_toolbar_button_glyph (w, tb);
	  
	  if (GLYPHP (glyph))
	    instance = glyph_image_instance (glyph, window, ERROR_ME_NOT, 1);
	  else
	    instance = Qnil;
	  
	  if (IMAGE_INSTANCEP (instance))
	    {
	      struct Lisp_Image_Instance* p = XIMAGE_INSTANCE (instance);
	      
	      if (IMAGE_INSTANCE_PIXMAP_TYPE_P (p))
		{
		  /* we are going to honour the toolbar settings and
		     resize the bitmaps accordingly */
		  
		  if (IMAGE_INSTANCE_PIXMAP_WIDTH (p) > bmwidth
		      ||
		      IMAGE_INSTANCE_PIXMAP_HEIGHT (p) > bmheight)
		    {
		      if (!mswindows_resize_dibitmap_instance 
			  (p, f, bmwidth, bmheight))
			{
			  xfree (button_tbl);
			  if (ilist) ImageList_Destroy (ilist);
			  signal_simple_error ("couldn't resize pixmap", 
					       instance);
			}
		    }
		  else 
		    {
		      bmwidth = IMAGE_INSTANCE_PIXMAP_WIDTH (p);
		      bmheight = IMAGE_INSTANCE_PIXMAP_HEIGHT (p);
		    }
	      
		  /* need to build an image list for the bitmaps */
		  if (!ilist)
		    {
		      if (!(ilist = ImageList_Create 
			    ( IMAGE_INSTANCE_PIXMAP_WIDTH (p),
			      IMAGE_INSTANCE_PIXMAP_HEIGHT (p),
			      ILC_COLOR24, 	
			      nbuttons,
			      nbuttons * 2 )))
			{
			  xfree (button_tbl);
			  signal_simple_error ("couldn't create image list",
					       instance);
			}
		    }
  
		  /* add a bitmap to the list */
		  if ((tbbutton->iBitmap =
		       ImageList_Add (ilist, 
				      IMAGE_INSTANCE_MSWINDOWS_BITMAP (p),
				      IMAGE_INSTANCE_MSWINDOWS_MASK (p))) < 0)
		    {
		      xfree (button_tbl);
		      if (ilist) ImageList_Destroy (ilist);
		      signal_simple_error ("image list creation failed", 
					   instance);
		    }
		}
	    }

	  Fputhash (make_int (tbbutton->idCommand), 
		    button, FRAME_MSWINDOWS_TOOLBAR_HASHTABLE (f));
      
	  tbbutton++;
	  button = tb->next;
	}

      button = FRAME_TOOLBAR_DATA (f, pos)->toolbar_buttons;

  /* create the toolbar window? */
      if (!toolbarwnd 
	  &&
	  (toolbarwnd = 
	   CreateWindowEx ( WS_EX_WINDOWEDGE,
			    TOOLBARCLASSNAME,
			    NULL,
			    WS_CHILD | WS_VISIBLE | WS_DLGFRAME | TBSTYLE_TOOLTIPS 
			    | CCS_NORESIZE | CCS_NOPARENTALIGN | CCS_NODIVIDER,
			    x, y, bar_width, bar_height,
			    FRAME_MSWINDOWS_HANDLE (f),
			    (HMENU)(TOOLBAR_ID_BIAS + pos),
			    NULL, 
			    NULL))==NULL)
	{
	  xfree (button_tbl);
	  ImageList_Destroy (ilist);
	  error ("couldn't create toolbar");
	}
#if 0
      SendMessage (toolbarwnd, TB_SETPADDING,
		   0, MAKELPARAM(border_width, border_width));
#endif
      /* finally populate with images */
      if (SendMessage (toolbarwnd, TB_BUTTONSTRUCTSIZE,
		       (WPARAM)sizeof(TBBUTTON), (LPARAM)0) == -1) 
	{
	  mswindows_clear_toolbar (f, pos, 0);
	  error ("couldn't set button structure size");
	}

      /* set the size of buttons */
      SendMessage (toolbarwnd, TB_SETBUTTONSIZE, 0, 
		   (LPARAM)MAKELONG (width, height));
		   
      /* set the size of bitmaps */
      SendMessage (toolbarwnd, TB_SETBITMAPSIZE, 0, 
		   (LPARAM)MAKELONG (bmwidth, bmheight));
		   
      /* finally populate with images */
      if (!SendMessage (toolbarwnd, TB_ADDBUTTONS,
			(WPARAM)nbuttons, (LPARAM)button_tbl))
	{
	  mswindows_clear_toolbar (f, pos, 0);
	  error ("couldn't add button list to toolbar");
	}

      /* vertical toolbars need more rows */
      if (vert)
	{
	  SendMessage (toolbarwnd, TB_SETROWS, 
		       MAKEWPARAM(nbuttons, FALSE), 0);
	}

      else
	{
	  SendMessage (toolbarwnd, TB_SETROWS, MAKEWPARAM(1, FALSE), 0);
	}

      /* finally populate with images */
      if (SendMessage (toolbarwnd, TB_SETIMAGELIST, 0,
		       (LPARAM)ilist) == -1) 
	{
	  mswindows_clear_toolbar (f, pos, 0);
	  error ("couldn't add image list to toolbar");
	}

      /* now display the window */
      ShowWindow (toolbarwnd, SW_SHOW);

      if (button_tbl) xfree (button_tbl);

      SET_TOOLBAR_WAS_VISIBLE_FLAG (f, pos, 1);
    }
}

static void
mswindows_move_toolbar (struct frame *f, enum toolbar_pos pos)
{
  int bar_x, bar_y, bar_width, bar_height, vert;
  HWND toolbarwnd = TOOLBAR_HANDLE(f,pos);
  
  if (toolbarwnd)
    {
      get_toolbar_coords (f, pos, &bar_x, &bar_y, &bar_width, &bar_height,
			  &vert, 1);

      /* #### This terrible mangling with coordinates perhaps
	 arises from different treatment of toolbar positions
	 by Windows and by XEmacs. */
      switch (pos)
	{
	case TOP_TOOLBAR:
	  bar_x -= 2; bar_y--;
	  bar_width += 2; bar_height++;
	  break;
	case LEFT_TOOLBAR:
	  bar_x -= 2; bar_y--;
	  bar_width++; bar_height++;
	  break;
	case BOTTOM_TOOLBAR:
	  bar_x--;
	  bar_width++;
	  break;
	case RIGHT_TOOLBAR:
	  bar_y--;
	  break;
	}
      SetWindowPos (toolbarwnd, NULL, bar_x, bar_y, 
		    bar_width + 1, bar_height + 1, SWP_NOZORDER);
    }
}

static void
mswindows_redraw_exposed_toolbars (struct frame *f, int x, int y, int width,
				   int height)
{
  assert (FRAME_MSWINDOWS_P (f));

  if (FRAME_REAL_TOP_TOOLBAR_VISIBLE (f))
    mswindows_move_toolbar (f, TOP_TOOLBAR);

  if (FRAME_REAL_BOTTOM_TOOLBAR_VISIBLE (f))
    mswindows_move_toolbar (f, BOTTOM_TOOLBAR);

  if (FRAME_REAL_LEFT_TOOLBAR_VISIBLE (f))
    mswindows_move_toolbar (f, LEFT_TOOLBAR);

  if (FRAME_REAL_RIGHT_TOOLBAR_VISIBLE (f))
    mswindows_move_toolbar (f, RIGHT_TOOLBAR);
}

static void
mswindows_initialize_frame_toolbars (struct frame *f)
{

}

static void
mswindows_output_frame_toolbars (struct frame *f)
{
  assert (FRAME_MSWINDOWS_P (f));

  if (FRAME_REAL_TOP_TOOLBAR_VISIBLE (f))
    mswindows_output_toolbar (f, TOP_TOOLBAR);
  else if (f->top_toolbar_was_visible)
    mswindows_clear_toolbar (f, TOP_TOOLBAR, 0);

  if (FRAME_REAL_BOTTOM_TOOLBAR_VISIBLE (f))
    mswindows_output_toolbar (f, BOTTOM_TOOLBAR);
  else if (f->bottom_toolbar_was_visible)
    mswindows_clear_toolbar (f, BOTTOM_TOOLBAR, 0);

  if (FRAME_REAL_LEFT_TOOLBAR_VISIBLE (f))
    mswindows_output_toolbar (f, LEFT_TOOLBAR);
  else if (f->left_toolbar_was_visible)
    mswindows_clear_toolbar (f, LEFT_TOOLBAR, 0);

  if (FRAME_REAL_RIGHT_TOOLBAR_VISIBLE (f))
    mswindows_output_toolbar (f, RIGHT_TOOLBAR);
  else if (f->right_toolbar_was_visible)
    mswindows_clear_toolbar (f, RIGHT_TOOLBAR, 0);
}

static void
mswindows_free_frame_toolbars (struct frame *f)
{
  HWND twnd=NULL;
#define DELETE_TOOLBAR(pos) \
  mswindows_clear_toolbar(f, 0, pos); \
  if ((twnd=GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS + pos))) \
      DestroyWindow(twnd)

  DELETE_TOOLBAR(TOP_TOOLBAR);
  DELETE_TOOLBAR(BOTTOM_TOOLBAR);
  DELETE_TOOLBAR(LEFT_TOOLBAR);
  DELETE_TOOLBAR(RIGHT_TOOLBAR);
#undef DELETE_TOOLBAR
}

/* map toolbar hwnd to pos*/
int mswindows_find_toolbar_pos(struct frame* f, HWND ctrl)
{
#if 1
  int id = GetDlgCtrlID(ctrl);
  return id ? id - TOOLBAR_ID_BIAS : -1;
#else
  if (GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS) == ctrl)
    return 0;
  else if (GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS +1) == ctrl)
    return 1;
  else if (GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS +2) == ctrl)
    return 2;
  else if (GetDlgItem(FRAME_MSWINDOWS_HANDLE(f), TOOLBAR_ID_BIAS +3) == ctrl)
    return 3;
  else
    assert(0);
#endif
}

Lisp_Object 
mswindows_get_toolbar_button_text ( struct frame* f, int command_id )
{
  Lisp_Object button = Fgethash (make_int (command_id),
				 FRAME_MSWINDOWS_TOOLBAR_HASHTABLE (f), Qnil);
  
  if (!NILP (button))
    {
      struct toolbar_button *tb = XTOOLBAR_BUTTON (button);
      return tb->help_string;
    }
  return Qnil;
}

/*
 * Return value is Qt if we have dispatched the command,
 * or Qnil if id has not been mapped to a callback.
 * Window procedure may try other targets to route the
 * command if we return nil
 */
Lisp_Object
mswindows_handle_toolbar_wm_command (struct frame* f, HWND ctrl, WORD id)
{
  /* Try to map the command id through the proper hash table */
  Lisp_Object button, command, funcsym, frame;
  struct gcpro gcpro1;
  
  button = Fgethash (make_int (id), 
		     FRAME_MSWINDOWS_TOOLBAR_HASHTABLE (f), Qnil);

  if (NILP (button))
    return Qnil;

  command = XTOOLBAR_BUTTON(button)->callback;
  
  if (UNBOUNDP(command))
    return Qnil;
  
  /* Need to gcpro because the hashtable may get destroyed
     by menu_cleanup(), and will not gcpro the command
     any more */
  GCPRO1 (command);
  
  /* Ok, this is our one. Enqueue it. */
  if (SYMBOLP (command))
      funcsym = Qcall_interactively;
  else if (CONSP (command))
      funcsym = Qeval;
  else
    signal_simple_error ("Callback must be either evallable form or a symbol",
			 command);

  XSETFRAME (frame, f);
  enqueue_misc_user_event (frame, funcsym, command);

  /* Needs good bump also, for WM_COMMAND may have been dispatched from
     mswindows_need_event, which will block again despite new command
     event has arrived */
  mswindows_bump_queue ();
  
  UNGCPRO; /* command */
  return Qt;
}


/************************************************************************/
/*                            initialization                            */
/************************************************************************/

void
console_type_create_toolbar_mswindows (void)
{
  CONSOLE_HAS_METHOD (mswindows, output_frame_toolbars);
  CONSOLE_HAS_METHOD (mswindows, initialize_frame_toolbars);
  CONSOLE_HAS_METHOD (mswindows, free_frame_toolbars);
  CONSOLE_HAS_METHOD (mswindows, redraw_exposed_toolbars);
}