/* GTK - The GIMP Toolkit
 * Copyright (C) 2010 Carlos Garnacho <carlosg@gnome.org>
 * Copyright (C) 2011 Red Hat, Inc.
 *
 * Authors: Carlos Garnacho <carlosg@gnome.org>
 *          Cosimo Cecchi <cosimoc@gnome.org>
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the
 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 */

#include <config.h>

#include "gtkwin32themeprivate.h"

#ifdef G_OS_WIN32

#include <windows.h>
#include <cairo-win32.h>

typedef HANDLE HTHEME;

#define UXTHEME_DLL "uxtheme.dll"

static HINSTANCE uxtheme_dll = NULL;
static gboolean use_xp_theme = FALSE;

typedef HRESULT (FAR PASCAL *GetThemeSysFontFunc)           (HTHEME hTheme, int iFontID, OUT LOGFONTW *plf);
typedef int (FAR PASCAL *GetThemeSysSizeFunc)               (HTHEME hTheme, int iSizeId);
typedef COLORREF (FAR PASCAL *GetThemeSysColorFunc)         (HTHEME hTheme,
							     int iColorID);
typedef HTHEME (FAR PASCAL *OpenThemeDataFunc)              (HWND hwnd,
							     LPCWSTR pszClassList);
typedef HRESULT (FAR PASCAL *CloseThemeDataFunc)            (HTHEME theme);
typedef HRESULT (FAR PASCAL *DrawThemeBackgroundFunc)       (HTHEME hTheme, HDC hdc, int iPartId, int iStateId,
							     const RECT *pRect, const RECT *pClipRect);
typedef HRESULT (FAR PASCAL *EnableThemeDialogTextureFunc)  (HWND hwnd,
							     DWORD dwFlags);
typedef BOOL (FAR PASCAL *IsThemeActiveFunc)                (VOID);
typedef BOOL (FAR PASCAL *IsAppThemedFunc)                  (VOID);
typedef BOOL (FAR PASCAL *IsThemeBackgroundPartiallyTransparentFunc) (HTHEME hTheme,
								      int iPartId,
								      int iStateId);
typedef HRESULT (FAR PASCAL *DrawThemeParentBackgroundFunc) (HWND hwnd,
							     HDC hdc,
							     RECT *prc);
typedef HRESULT (FAR PASCAL *GetThemePartSizeFunc)          (HTHEME hTheme,
							     HDC hdc,
							     int iPartId,
							     int iStateId,
							     RECT *prc,
							     int eSize,
							     SIZE *psz);

static GetThemeSysFontFunc get_theme_sys_font = NULL;
static GetThemeSysColorFunc get_theme_sys_color = NULL;
static GetThemeSysSizeFunc get_theme_sys_metric = NULL;
static OpenThemeDataFunc open_theme_data = NULL;
static CloseThemeDataFunc close_theme_data = NULL;
static DrawThemeBackgroundFunc draw_theme_background = NULL;
static EnableThemeDialogTextureFunc enable_theme_dialog_texture = NULL;
static IsThemeActiveFunc is_theme_active = NULL;
static IsAppThemedFunc is_app_themed = NULL;
static IsThemeBackgroundPartiallyTransparentFunc is_theme_partially_transparent = NULL;
static DrawThemeParentBackgroundFunc draw_theme_parent_background = NULL;
static GetThemePartSizeFunc get_theme_part_size = NULL;

static GHashTable *hthemes_by_class = NULL;

static void
_gtk_win32_theme_init (void)
{
  char *buf;
  char dummy;
  int n, k;

  if (uxtheme_dll)
    return;

  n = GetSystemDirectory (&dummy, 0);
  if (n <= 0)
    return;

  buf = g_malloc (n + 1 + strlen (UXTHEME_DLL));
  k = GetSystemDirectory (buf, n);
  if (k == 0 || k > n)
    {
      g_free (buf);
      return;
    }

  if (!G_IS_DIR_SEPARATOR (buf[strlen (buf) -1]))
    strcat (buf, G_DIR_SEPARATOR_S);
  strcat (buf, UXTHEME_DLL);

  uxtheme_dll = LoadLibrary (buf);
  g_free (buf);

  if (!uxtheme_dll)
    return;

  is_app_themed = (IsAppThemedFunc) GetProcAddress (uxtheme_dll, "IsAppThemed");
  if (is_app_themed)
    {
      is_theme_active = (IsThemeActiveFunc) GetProcAddress (uxtheme_dll, "IsThemeActive");
      open_theme_data = (OpenThemeDataFunc) GetProcAddress (uxtheme_dll, "OpenThemeData");
      close_theme_data = (CloseThemeDataFunc) GetProcAddress (uxtheme_dll, "CloseThemeData");
      draw_theme_background = (DrawThemeBackgroundFunc) GetProcAddress (uxtheme_dll, "DrawThemeBackground");
      enable_theme_dialog_texture = (EnableThemeDialogTextureFunc) GetProcAddress (uxtheme_dll, "EnableThemeDialogTexture");
      get_theme_sys_font = (GetThemeSysFontFunc) GetProcAddress (uxtheme_dll, "GetThemeSysFont");
      get_theme_sys_color = (GetThemeSysColorFunc) GetProcAddress (uxtheme_dll, "GetThemeSysColor");
      get_theme_sys_metric = (GetThemeSysSizeFunc) GetProcAddress (uxtheme_dll, "GetThemeSysSize");
      is_theme_partially_transparent = (IsThemeBackgroundPartiallyTransparentFunc) GetProcAddress (uxtheme_dll, "IsThemeBackgroundPartiallyTransparent");
      draw_theme_parent_background = (DrawThemeParentBackgroundFunc) GetProcAddress (uxtheme_dll, "DrawThemeParentBackground");
      get_theme_part_size = (GetThemePartSizeFunc) GetProcAddress (uxtheme_dll, "GetThemePartSize");
    }

  if (is_app_themed && is_theme_active)
    {
      use_xp_theme = (is_app_themed () && is_theme_active ());
    }
  else
    {
      use_xp_theme = FALSE;
    }

  hthemes_by_class = g_hash_table_new (g_str_hash, g_str_equal);
}

static HTHEME
lookup_htheme_by_classname (const char *class)
{
  HTHEME theme;
  guint16 *wclass;
  char *lower;
  
  lower = g_ascii_strdown (class, -1);

  theme = (HTHEME)  g_hash_table_lookup (hthemes_by_class, lower);
  if (theme)
    {
      g_free (lower);
      return theme;
    }

  wclass = g_utf8_to_utf16 (lower, -1, NULL, NULL, NULL);
  theme  = open_theme_data (NULL, wclass);
  g_free (wclass);

  if (theme == NULL)
    {
      g_free (lower);
      return NULL;
    }

  /* Takes ownership of lower: */
  g_hash_table_insert (hthemes_by_class, lower, theme);

  return theme;
}

#else

typedef void * HTHEME;

static void
_gtk_win32_theme_init (void)
{
}

static HTHEME
lookup_htheme_by_classname (const char *class)
{
  return NULL;
}

#endif /* G_OS_WIN32 */

G_DEFINE_BOXED_TYPE_WITH_CODE (GtkWin32ThemePart, _gtk_win32_theme_part,
			       _gtk_win32_theme_part_ref, _gtk_win32_theme_part_unref, 
			       _gtk_win32_theme_init() )

struct _GtkWin32ThemePart {
  HTHEME theme;
  int part;
  int state;

  double over_alpha;
  int part2;
  int state2;

  gint margins[4];

  gint ref_count;
};

GtkWin32ThemePart *
_gtk_win32_theme_part_new (const char *class, 
			   int xp_part, int state, 
			   int xp_part2, int state2, 
			   double over_alpha,
			   gint margins[4])
{
  GtkWin32ThemePart *part;
  int i;

  part = g_slice_new0 (GtkWin32ThemePart);
  part->ref_count = 1;

  part->theme = lookup_htheme_by_classname (class);
  part->part = xp_part;
  part->state = state;
  part->part2 = xp_part2;
  part->state2 = state2;
  part->over_alpha = over_alpha;
  for (i = 0; i < 4; i++)
    part->margins[i] = margins[i];

  return part;
}

GtkWin32ThemePart *
_gtk_win32_theme_part_ref (GtkWin32ThemePart *part)
{
  g_return_val_if_fail (part != NULL, NULL);

  part->ref_count++;

  return part;
}

void
_gtk_win32_theme_part_unref (GtkWin32ThemePart *part)
{
  g_return_if_fail (part != NULL);

  part->ref_count--;

  if (part->ref_count == 0)
    {
      g_slice_free (GtkWin32ThemePart, part);
    }
}

int
_gtk_win32_theme_part_parse (GtkCssParser *parser, 
			     GFile *base, 
			     GValue *value)
{
  char *class;
  int xp_part, state, xp_part2, state2;
  double over_alpha;
  GtkWin32ThemePart *theme_part;
  gint i, margins[4];

  if (!_gtk_css_parser_try (parser, "-gtk-win32-theme-part", TRUE))
    {
      return -1;
    }
  
  g_value_unset (value);
  g_value_init (value, GTK_TYPE_WIN32_THEME_PART);

  if (!_gtk_css_parser_try (parser, "(", TRUE))
    {
      _gtk_css_parser_error (parser,
                             "Expected '(' after '-gtk-win32-theme-part'");
      return 0;
    }
  
  class = _gtk_css_parser_try_name (parser, TRUE);
  if (class == NULL)
    {
      _gtk_css_parser_error (parser,
                             "Expected name as first argument to  '-gtk-win32-theme-part'");
      return 0;
    }

  if (! _gtk_css_parser_try (parser, ",", TRUE))
    {
      g_free (class);
      _gtk_css_parser_error (parser,
			     "Expected ','");
      return 0;
    }

  if (!_gtk_css_parser_try_int (parser, &xp_part))
    {
      g_free (class);
      _gtk_css_parser_error (parser, "Expected a valid integer value");
      return 0;
    }

  if (!_gtk_css_parser_try_int (parser, &state))
    {
      g_free (class);
      _gtk_css_parser_error (parser, "Expected a valid integer value");
      return 0;
    }


  margins[0] = margins[1] = margins[2] = margins[3] = 0;
  over_alpha = 1.0;
  xp_part2 = -1;
  state2 = -1;

  while (TRUE)
    {
      if ( _gtk_css_parser_try (parser, ",", TRUE))
	{
	  if ( _gtk_css_parser_try (parser, "over", TRUE))
	    {
	      if (!_gtk_css_parser_try (parser, "(", TRUE))
		{
		  _gtk_css_parser_error (parser,
					 "Expected '(' after 'over'");
		  return 0;
		}

	      if (!_gtk_css_parser_try_int (parser, &xp_part2))
		{
		  g_free (class);
		  _gtk_css_parser_error (parser, "Expected a valid integer value");
		  return 0;
		}

	      if (!_gtk_css_parser_try_int (parser, &state2))
		{
		  g_free (class);
		  _gtk_css_parser_error (parser, "Expected a valid integer value");
		  return 0;
		}

	      if ( _gtk_css_parser_try (parser, ",", TRUE))
		{
		  if (!_gtk_css_parser_try_double (parser, &over_alpha))
		    {
		      g_free (class);
		      _gtk_css_parser_error (parser, "Expected a valid double value");
		      return 0;
		    }
		}

	      if (!_gtk_css_parser_try (parser, ")", TRUE))
		{
		  g_free (class);
		  _gtk_css_parser_error (parser,
					 "Expected ')' at end of 'over'");
		  return 0;
		}
	    }
	  else if ( _gtk_css_parser_try (parser, "margins", TRUE))
	    {
	      if (!_gtk_css_parser_try (parser, "(", TRUE))
		{
		  g_free (class);
		  _gtk_css_parser_error (parser,
					 "Expected '(' after 'margins'");
		  return 0;
		}

	      for (i = 0; i < 4; i++)
		{
		  if (!_gtk_css_parser_try_int (parser, &margins[i]))
		    break;
		}
	      
	      if (i == 0)
		{
		  g_free (class);
		  _gtk_css_parser_error (parser, "Expected valid margins");
		  return 0;
		}

	      if (i == 1)
		margins[1] = margins[0];
	      if (i <= 2)
		margins[2] = margins[1];
	      if (i <= 3)
		margins[3] = margins[2];
	      
	      if (!_gtk_css_parser_try (parser, ")", TRUE))
		{
		  g_free (class);
		  _gtk_css_parser_error (parser,
					 "Expected ')' at end of 'margins'");
		  return 0;
		}
	    }
	  else
	    {
	      _gtk_css_parser_error (parser,
				     "Expected identifier");
	      return 0;
	    }
	}
      else
	break; /* no comma, break loop */
    }

  if (!_gtk_css_parser_try (parser, ")", TRUE))
    {
      g_free (class);
      _gtk_css_parser_error (parser,
			     "Expected ')'");
      return 0;
    }
  
  theme_part = _gtk_win32_theme_part_new (class, 
					  xp_part, state, 
					  xp_part2, state2,
					  over_alpha,
					  margins);
  g_free (class);
  
  g_value_take_boxed (value, theme_part);
  return 1;
}

#ifdef G_OS_WIN32
cairo_surface_t *
_gtk_win32_theme_part_create_surface  (GtkWin32ThemePart  *part,
				       int                 xp_part,
				       int                 state,
				       int                 width,
				       int                 height)
{
  cairo_surface_t *surface;
  HDC hdc;
  RECT rect;
  HRESULT res;

  surface = cairo_win32_surface_create_with_dib (CAIRO_FORMAT_ARGB32, width, height);
  hdc = cairo_win32_surface_get_dc (surface);
  
  rect.left = part->margins[3];
  rect.top = part->margins[0];
  rect.right = width - part->margins[1];
  rect.bottom = height - part->margins[2];

  res = draw_theme_background (part->theme, hdc, xp_part, state, &rect, &rect);
  return surface;
}
#endif


cairo_pattern_t *
_gtk_win32_theme_part_render  (GtkWin32ThemePart  *part,
			       int                 width,
			       int                 height)
{
#ifdef G_OS_WIN32
  cairo_surface_t *surface, *surface2, *image;
  cairo_pattern_t *pattern;
  cairo_t *cr;
  cairo_matrix_t matrix;
  cairo_user_data_key_t key;

  surface = _gtk_win32_theme_part_create_surface  (part, part->part, part->state, 
						   width, height);
  
  if (part->state2 >= 0)
    {
      surface2 = _gtk_win32_theme_part_create_surface  (part, part->part2, part->state2, 
							width, height);


      cr = cairo_create (surface);

      pattern = cairo_pattern_create_for_surface (surface2);
      cairo_set_source (cr, pattern);
      cairo_set_operator (cr, CAIRO_OPERATOR_OVER);
      cairo_paint_with_alpha (cr, part->over_alpha);
      
      cairo_destroy (cr);
      cairo_pattern_destroy (pattern);

      cairo_surface_destroy (surface2);
    }

  /* We need to return an image surface, as that is what the code expects in order
     to get the size */
  image = cairo_win32_surface_get_image (surface);
  pattern = cairo_pattern_create_for_surface (cairo_surface_reference (image));

  cairo_matrix_init_scale (&matrix,
			   width,
			   height);
  cairo_pattern_set_matrix (pattern, &matrix);

  /* We can't immediately destroy the surface, because that would free the data
     the image surface refers too. Instead we destroy it with the pattern. */
  cairo_pattern_set_user_data (pattern,
			       &key,
			       surface, (cairo_destroy_func_t) cairo_surface_destroy);

  return pattern;
#else
  GdkRGBA color;
  
  gdk_rgba_parse (&color, "pink");

  return cairo_pattern_create_rgb (color.red, color.green, color.blue);
#endif
}

int
_gtk_win32_theme_int_parse (GtkCssParser      *parser,
			    GFile             *base,
			    int               *value)
{
  char *class;
  int arg;

  if (_gtk_css_parser_try (parser,
			   "-gtk-win32-size",
			   TRUE))
    {
      if (!_gtk_css_parser_try (parser, "(", TRUE))
	{
	  _gtk_css_parser_error (parser,
				 "Expected '(' after '-gtk-win32-size'");
	  return 0;
	}

      class = _gtk_css_parser_try_name (parser, TRUE);
      if (class == NULL)
	{
	  _gtk_css_parser_error (parser,
				 "Expected name as first argument to  '-gtk-win32-size'");
	  return 0;
	}

      if (! _gtk_css_parser_try (parser, ",", TRUE))
	{
	  g_free (class);
	  _gtk_css_parser_error (parser,
				 "Expected ','");
	  return 0;
	}

      if (!_gtk_css_parser_try_int (parser, &arg))
	{
	  g_free (class);
	  _gtk_css_parser_error (parser, "Expected a valid integer value");
	  return 0;
	}

      if (!_gtk_css_parser_try (parser, ")", TRUE))
	{
	  _gtk_css_parser_error (parser,
				 "Expected ')'");
	  return 0;
	}

#ifdef G_OS_WIN32
      if (use_xp_theme && get_theme_sys_metric != NULL)
	{
	  HTHEME theme = lookup_htheme_by_classname (class);

	  /* If theme is NULL it will just return the GetSystemMetrics value */
	  *value = get_theme_sys_metric (theme, arg);
	}
      else
	*value = GetSystemMetrics (arg);
#else
      *value = 1;
#endif

      g_free (class);

      return 1;
    }

  return -1;
}

GtkSymbolicColor *
_gtk_win32_theme_color_parse (GtkCssParser *parser)
{
  GtkSymbolicColor *color;
  char *class;
  int id;

  class = _gtk_css_parser_try_name (parser, TRUE);
  if (class == NULL)
    {
      _gtk_css_parser_error (parser,
			     "Expected name as first argument to  '-gtk-win32-color'");
      return NULL;
    }

  if (! _gtk_css_parser_try (parser, ",", TRUE))
    {
      g_free (class);
      _gtk_css_parser_error (parser,
			     "Expected ','");
      return NULL;
    }

  if (!_gtk_css_parser_try_int (parser, &id))
    {
      g_free (class);
      _gtk_css_parser_error (parser, "Expected a valid integer value");
      return NULL;
    }

  color = gtk_symbolic_color_new_win32 (class, id);
  g_free (class);
  return color;
}

gboolean
_gtk_win32_theme_color_resolve (const char *theme_class,
				gint id,
				GdkRGBA *color)
{
#ifdef G_OS_WIN32
  DWORD dcolor;

  if (use_xp_theme && get_theme_sys_color != NULL)
    {
      HTHEME theme = lookup_htheme_by_classname (theme_class);

      /* if theme is NULL, it will just return the GetSystemColor()
         value */
      dcolor = get_theme_sys_color (theme, id);
    }
  else
    dcolor = GetSysColor (id);

  color->alpha = 1.0;
  color->red = GetRValue (dcolor) / 255.0;
  color->green = GetGValue (dcolor) / 255.0;
  color->blue = GetBValue (dcolor) / 255.0;
#else
  gdk_rgba_parse (color, "pink");
#endif
  return TRUE;
}