/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
 *
 * Copyright (C) 2004-2009 William Jon McCann <mccann@jhu.edu>
 * Copyright (C) 2009      Red Hat, Inc.
 *
 * This program 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 of the License, or
 * (at your option) any later version.
 *
 * This program 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 this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.
 *
 * Authors: William Jon McCann <mccann@jhu.edu>
 *
 */

#include "config.h"

#include <stdlib.h>
#include <stdio.h>
#include <time.h>
#include <errno.h>

#include <string.h>
#include <sys/time.h>
#include <sys/types.h>
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif /* HAVE_UNISTD_H */

#include <gdk/gdkx.h>
#include <gtk/gtk.h>

#include "gs-fade.h"
#include "gs-debug.h"

#define MATE_DESKTOP_USE_UNSTABLE_API

#include "libmate-desktop/mate-rr.h"

/* XFree86 4.x+ Gamma fading */


#ifdef HAVE_XF86VMODE_GAMMA

#include <X11/extensions/xf86vmode.h>

#define XF86_MIN_GAMMA  0.1

#endif /* HAVE_XF86VMODE_GAMMA */

static void     gs_fade_class_init (GSFadeClass *klass);
static void     gs_fade_init       (GSFade      *fade);
static void     gs_fade_finalize   (GObject        *object);

#define GS_FADE_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GS_TYPE_FADE, GSFadePrivate))

struct GSGammaInfo
{
	int              size;
	unsigned short  *r;
	unsigned short  *g;
	unsigned short  *b;
};

struct GSFadeScreenPrivate
{
	int                 fade_type;
	int                 num_ramps;
	/* one per crtc in randr mode */
	struct GSGammaInfo *info;
	/* one per screen in theory */
	MateRRScreen      *rrscreen;
#ifdef HAVE_XF86VMODE_GAMMA
	/* one per screen also */
	XF86VidModeGamma    vmg;
#endif /* HAVE_XF86VMODE_GAMMA */
	gboolean (*fade_setup)           (GSFade *fade,
	                                  int     screen);
	gboolean (*fade_set_alpha_gamma) (GSFade *fade,
	                                  int     screen,
	                                  gdouble alpha);
	void     (*fade_finish)          (GSFade *fade,
	                                  int     screen);
};

struct GSFadePrivate
{
	guint            enabled : 1;
	guint            active : 1;

	guint            timeout;

	guint            step;
	guint            num_steps;
	guint            timer_id;

	gdouble          alpha_per_iter;
	gdouble          current_alpha;

	int              num_screens;

	struct GSFadeScreenPrivate *screen_priv;
};

enum
{
    FADED,
    LAST_SIGNAL
};

enum
{
    FADE_TYPE_NONE,
    FADE_TYPE_GAMMA_NUMBER,
    FADE_TYPE_GAMMA_RAMP,
    FADE_TYPE_XRANDR,
};

static guint         signals [LAST_SIGNAL] = { 0, };

G_DEFINE_TYPE (GSFade, gs_fade, G_TYPE_OBJECT)

static gpointer fade_object = NULL;

#ifdef HAVE_XF86VMODE_GAMMA

/* This is needed because the VidMode extension doesn't work
   on remote displays -- but if the remote display has the extension
   at all, XF86VidModeQueryExtension returns true, and then
   XF86VidModeQueryVersion dies with an X error.
*/

static gboolean error_handler_hit = FALSE;

static int
ignore_all_errors_ehandler (Display     *dpy,
                            XErrorEvent *error)
{
	error_handler_hit = TRUE;

	return 0;
}

static Bool
safe_XF86VidModeQueryVersion (Display *dpy,
                              int     *majP,
                              int     *minP)
{
	Bool          result;
	XErrorHandler old_handler;

	XSync (dpy, False);
	error_handler_hit = FALSE;
	old_handler = XSetErrorHandler (ignore_all_errors_ehandler);

	result = XF86VidModeQueryVersion (dpy, majP, minP);

	XSync (dpy, False);
	XSetErrorHandler (old_handler);
	XSync (dpy, False);

	return (error_handler_hit
	        ? False
	        : result);
}

static gboolean
xf86_whack_gamma (int              screen,
                  struct GSFadeScreenPrivate *screen_priv,
                  float            ratio)
{
	Bool status;
	struct GSGammaInfo *gamma_info;

	gamma_info = screen_priv->info;

	if (!gamma_info)
		return FALSE;

	if (ratio < 0)
	{
		ratio = 0;
	}
	if (ratio > 1)
	{
		ratio = 1;
	}

	if (gamma_info->size == 0)
	{
		/* we only have a gamma number, not a ramp. */

		XF86VidModeGamma g2;

		g2.red   = screen_priv->vmg.red   * ratio;
		g2.green = screen_priv->vmg.green * ratio;
		g2.blue  = screen_priv->vmg.blue  * ratio;

		if (g2.red < XF86_MIN_GAMMA)
		{
			g2.red = XF86_MIN_GAMMA;
		}
		if (g2.green < XF86_MIN_GAMMA)
		{
			g2.green = XF86_MIN_GAMMA;
		}
		if (g2.blue < XF86_MIN_GAMMA)
		{
			g2.blue = XF86_MIN_GAMMA;
		}

#if GTK_CHECK_VERSION (3, 0, 0)
		status = XF86VidModeSetGamma (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), screen, &g2);
#else
		status = XF86VidModeSetGamma (GDK_DISPLAY (), screen, &g2);
#endif
	}
	else
	{

# ifdef HAVE_XF86VMODE_GAMMA_RAMP
		unsigned short *r, *g, *b;
		int i;

		r = g_new0 (unsigned short, gamma_info->size);
		g = g_new0 (unsigned short, gamma_info->size);
		b = g_new0 (unsigned short, gamma_info->size);

		for (i = 0; i < gamma_info->size; i++)
		{
			r[i] = gamma_info->r[i] * ratio;
			g[i] = gamma_info->g[i] * ratio;
			b[i] = gamma_info->b[i] * ratio;
		}

#if GTK_CHECK_VERSION (3, 0, 0)
		status = XF86VidModeSetGammaRamp (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), screen, gamma_info->size, r, g, b);
#else
		status = XF86VidModeSetGammaRamp (GDK_DISPLAY (), screen, gamma_info->size, r, g, b);
#endif

		g_free (r);
		g_free (g);
		g_free (b);

# else  /* !HAVE_XF86VMODE_GAMMA_RAMP */
		abort ();
# endif /* !HAVE_XF86VMODE_GAMMA_RAMP */
	}

	gdk_flush ();

	return status;
}

#endif /* HAVE_XF86VMODE_GAMMA */

/* VidModeExtension version 2.0 or better is needed to do gamma.
   2.0 added gamma values; 2.1 added gamma ramps.
*/
# define XF86_VIDMODE_GAMMA_MIN_MAJOR 2
# define XF86_VIDMODE_GAMMA_MIN_MINOR 0
# define XF86_VIDMODE_GAMMA_RAMP_MIN_MAJOR 2
# define XF86_VIDMODE_GAMMA_RAMP_MIN_MINOR 1


gboolean
gs_fade_get_enabled (GSFade *fade)
{
	g_return_val_if_fail (GS_IS_FADE (fade), FALSE);

	return fade->priv->enabled;
}

void
gs_fade_set_enabled (GSFade  *fade,
                     gboolean enabled)
{
	g_return_if_fail (GS_IS_FADE (fade));

	if (fade->priv->enabled != enabled)
	{
		fade->priv->enabled = enabled;
	}
}

#ifdef HAVE_XF86VMODE_GAMMA
static gboolean
gamma_fade_setup (GSFade *fade, int screen_idx)
{
	gboolean         res;
	struct GSFadeScreenPrivate *screen_priv;

	screen_priv = &fade->priv->screen_priv[screen_idx];

	if (screen_priv->info)
		return TRUE;

# ifndef HAVE_XF86VMODE_GAMMA_RAMP
	if (FADE_TYPE_GAMMA_RAMP == screen_priv->fade_type)
	{
		/* server is newer than client! */
		screen_priv->fade_type = FADE_TYPE_GAMMA_NUMBER;
	}
# endif

# ifdef HAVE_XF86VMODE_GAMMA_RAMP

	screen_priv->info = g_new0(struct GSGammaInfo, 1);
	screen_priv->num_ramps = 1;

	if (FADE_TYPE_GAMMA_RAMP == screen_priv->fade_type)
	{
		/* have ramps */

#if GTK_CHECK_VERSION (3, 0, 0)
		res = XF86VidModeGetGammaRampSize (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), screen_idx, &screen_priv->info->size);
#else
		res = XF86VidModeGetGammaRampSize (GDK_DISPLAY (), screen_idx, &screen_priv->info->size);
#endif
		if (!res || screen_priv->info->size <= 0)
		{
			screen_priv->fade_type = FADE_TYPE_GAMMA_NUMBER;
			goto test_number;
		}

		screen_priv->info->r = g_new0 (unsigned short, screen_priv->info->size);
		screen_priv->info->g = g_new0 (unsigned short, screen_priv->info->size);
		screen_priv->info->b = g_new0 (unsigned short, screen_priv->info->size);

		if (! (screen_priv->info->r && screen_priv->info->g && screen_priv->info->b))
		{
			screen_priv->fade_type = FADE_TYPE_GAMMA_NUMBER;
			goto test_number;
		}

#if GTK_CHECK_VERSION (3, 0, 0)
		res = XF86VidModeGetGammaRamp (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()),
#else
		res = XF86VidModeGetGammaRamp (GDK_DISPLAY (),
#endif
		                               screen_idx,
		                               screen_priv->info->size,
		                               screen_priv->info->r,
		                               screen_priv->info->g,
		                               screen_priv->info->b);
		if (! res)
		{
			screen_priv->fade_type = FADE_TYPE_GAMMA_NUMBER;
			goto test_number;
		}
		gs_debug ("Initialized gamma ramp fade");
	}
# endif /* HAVE_XF86VMODE_GAMMA_RAMP */

test_number:
	if (FADE_TYPE_GAMMA_NUMBER == screen_priv->fade_type)
	{
		/* only have gamma parameter, not ramps. */

#if GTK_CHECK_VERSION (3, 0, 0)
		res = XF86VidModeGetGamma (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), screen_idx, &screen_priv->vmg);
#else
		res = XF86VidModeGetGamma (GDK_DISPLAY (), screen_idx, &screen_priv->vmg);
#endif
		if (! res)
		{
			screen_priv->fade_type = FADE_TYPE_NONE;
			goto test_none;
		}
		gs_debug ("Initialized gamma fade for screen %d: %f %f %f",
		          screen_idx,
		          screen_priv->vmg.red,
		          screen_priv->vmg.green,
		          screen_priv->vmg.blue);
	}

test_none:
	if (FADE_TYPE_NONE == screen_priv->fade_type)
	{
		goto FAIL;
	}

	return TRUE;
FAIL:

	return FALSE;
}
#endif /* HAVE_XF86VMODE_GAMMA */

static void
screen_fade_finish (GSFade *fade, int screen_idx)
{
	struct GSFadeScreenPrivate *screen_priv;
	int i;
	screen_priv = &fade->priv->screen_priv[screen_idx];

	if (!screen_priv->info)
		return;

	for (i = 0; i < screen_priv->num_ramps; i++)
	{
		if (screen_priv->info[i].r)
			g_free (screen_priv->info[i].r);
		if (screen_priv->info[i].g)
			g_free (screen_priv->info[i].g);
		if (screen_priv->info[i].b)
			g_free (screen_priv->info[i].b);
	}

	g_free (screen_priv->info);
	screen_priv->info = NULL;
	screen_priv->num_ramps = 0;
}

#ifdef HAVE_XF86VMODE_GAMMA
static gboolean
gamma_fade_set_alpha_gamma (GSFade *fade,
                            int screen_idx,
                            gdouble alpha)
{
	struct GSFadeScreenPrivate *screen_priv;
	gboolean res;

	screen_priv = &fade->priv->screen_priv[screen_idx];
	res = xf86_whack_gamma (screen_idx, screen_priv, alpha);

	return TRUE;
}
#endif /* HAVE_XF86VMODE_GAMMA */

static void
check_gamma_extension (GSFade *fade, int screen_idx)
{
	struct GSFadeScreenPrivate *screen_priv;
#ifdef HAVE_XF86VMODE_GAMMA
	int      event;
	int      error;
	int      major;
	int      minor;
	gboolean res;
#endif /* HAVE_XF86VMODE_GAMMA */

	screen_priv = &fade->priv->screen_priv[screen_idx];

#ifdef HAVE_XF86VMODE_GAMMA
#if GTK_CHECK_VERSION (3, 0, 0)
	res = XF86VidModeQueryExtension (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), &event, &error);
#else
	res = XF86VidModeQueryExtension (GDK_DISPLAY (), &event, &error);
#endif
	if (! res)
		goto fade_none;

#if GTK_CHECK_VERSION (3, 0, 0)
	res = safe_XF86VidModeQueryVersion (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()), &major, &minor);
#else
	res = safe_XF86VidModeQueryVersion (GDK_DISPLAY (), &major, &minor);
#endif
	if (! res)
		goto fade_none;

	if (major < XF86_VIDMODE_GAMMA_MIN_MAJOR ||
	        (major == XF86_VIDMODE_GAMMA_MIN_MAJOR &&
	         minor < XF86_VIDMODE_GAMMA_MIN_MINOR))
		goto fade_none;

	screen_priv->fade_setup = gamma_fade_setup;
	screen_priv->fade_finish = screen_fade_finish;
	screen_priv->fade_set_alpha_gamma = gamma_fade_set_alpha_gamma;

	if (major < XF86_VIDMODE_GAMMA_RAMP_MIN_MAJOR ||
	        (major == XF86_VIDMODE_GAMMA_RAMP_MIN_MAJOR &&
	         minor < XF86_VIDMODE_GAMMA_RAMP_MIN_MINOR))
	{
		screen_priv->fade_type = FADE_TYPE_GAMMA_NUMBER;
		return;
	}

	/* Copacetic */
	screen_priv->fade_type = FADE_TYPE_GAMMA_RAMP;
	return;
fade_none:
#endif
	screen_priv->fade_type = FADE_TYPE_NONE;
}

/* Xrandr support */

static gboolean xrandr_fade_setup (GSFade *fade, int screen_idx)
{
	struct GSFadeScreenPrivate *screen_priv;
	MateRRCrtc *crtc;
	MateRRCrtc **crtcs;
	int crtc_count = 0;
	struct GSGammaInfo *info;
	gboolean res;

	screen_priv = &fade->priv->screen_priv[screen_idx];

	if (screen_priv->info)
		return TRUE;

	/* refresh the screen info */
	mate_rr_screen_refresh (screen_priv->rrscreen, NULL);

	crtcs = mate_rr_screen_list_crtcs (screen_priv->rrscreen);
	while (*crtcs)
	{
		crtc_count++;
		crtcs++;
	};

	screen_priv->info = g_new0 (struct GSGammaInfo, crtc_count);
	screen_priv->num_ramps = crtc_count;

	crtc_count = 0;
	crtcs = mate_rr_screen_list_crtcs (screen_priv->rrscreen);
	while (*crtcs)
	{
		crtc = *crtcs;

		info = &screen_priv->info[crtc_count];

		/* if no mode ignore crtc */
		if (!mate_rr_crtc_get_current_mode (crtc))
		{
			info->size = 0;
			info->r = NULL;
			info->g = NULL;
			info->b = NULL;
		}
		else
		{
			res = mate_rr_crtc_get_gamma (crtc, &info->size,
			                              &info->r, &info->g,
			                              &info->b);
			if (res == FALSE)
				goto fail;
		}

		crtcs++;
		crtc_count++;
	}
	return TRUE;
fail:
	return FALSE;
}

static void xrandr_crtc_whack_gamma (MateRRCrtc *crtc,
                                     struct GSGammaInfo *gamma_info,
                                     float            ratio)
{
	unsigned short *r, *g, *b;
	int i;

	if (gamma_info->size == 0)
		return;

	if (ratio < 0)
	{
		ratio = 0;
	}
	if (ratio > 1)
	{
		ratio = 1;
	}

	r = g_new0 (unsigned short, gamma_info->size);
	g = g_new0 (unsigned short, gamma_info->size);
	b = g_new0 (unsigned short, gamma_info->size);

	for (i = 0; i < gamma_info->size; i++)
	{
		r[i] = gamma_info->r[i] * ratio;
		g[i] = gamma_info->g[i] * ratio;
		b[i] = gamma_info->b[i] * ratio;
	}

	mate_rr_crtc_set_gamma (crtc, gamma_info->size,
	                        r, g, b);
	g_free (r);
	g_free (g);
	g_free (b);
}

static gboolean xrandr_fade_set_alpha_gamma (GSFade *fade,
        int screen_idx,
        gdouble alpha)
{
	struct GSFadeScreenPrivate *screen_priv;
	struct GSGammaInfo *info;
	MateRRCrtc **crtcs;
	int i;

	screen_priv = &fade->priv->screen_priv[screen_idx];

	if (!screen_priv->info)
		return FALSE;

	crtcs = mate_rr_screen_list_crtcs (screen_priv->rrscreen);
	i = 0;

	while (*crtcs)
	{
		info = &screen_priv->info[i];
		xrandr_crtc_whack_gamma (*crtcs, info, alpha);
		i++;
		crtcs++;
	}
	return TRUE;
}

static void
check_randr_extension (GSFade *fade, int screen_idx)
{
	GdkDisplay *display = gdk_display_get_default ();
	GdkScreen *screen = gdk_display_get_screen (display, screen_idx);
	struct GSFadeScreenPrivate *screen_priv;

	screen_priv = &fade->priv->screen_priv[screen_idx];

	screen_priv->rrscreen = mate_rr_screen_new (screen,
	                        NULL,
	                        NULL,
	                        NULL);
	if (!screen_priv->rrscreen)
	{
		screen_priv->fade_type = FADE_TYPE_NONE;
		return;
	}

	screen_priv->fade_type = FADE_TYPE_XRANDR;
	screen_priv->fade_setup = xrandr_fade_setup;
	screen_priv->fade_finish = screen_fade_finish;
	screen_priv->fade_set_alpha_gamma = xrandr_fade_set_alpha_gamma;
}

static gboolean
gs_fade_set_alpha (GSFade *fade,
                   gdouble alpha)
{
	gboolean ret = FALSE;
	int i;

	for (i = 0; i < fade->priv->num_screens; i++)
	{
		switch (fade->priv->screen_priv[i].fade_type)
		{
		case FADE_TYPE_GAMMA_RAMP:
		case FADE_TYPE_GAMMA_NUMBER:
		case FADE_TYPE_XRANDR:
			ret = fade->priv->screen_priv[i].fade_set_alpha_gamma (fade, i, alpha);
			break;
		case FADE_TYPE_NONE:
			ret = FALSE;
			break;
		default:
			g_warning ("Unknown fade type");
			ret = FALSE;
			break;
		}
	}
	return ret;
}

static gboolean
gs_fade_out_iter (GSFade *fade)
{
	gboolean ret;

	if (fade->priv->current_alpha < 0.01)
	{
		return FALSE;
	}

	fade->priv->current_alpha -= fade->priv->alpha_per_iter;

	ret = gs_fade_set_alpha (fade, fade->priv->current_alpha);

	return ret;
}

static gboolean
gs_fade_stop (GSFade *fade)
{
	if (fade->priv->timer_id > 0)
	{
		g_source_remove (fade->priv->timer_id);
		fade->priv->timer_id = 0;
	}

	fade->priv->step = 0;
	fade->priv->active = FALSE;

	return TRUE;
}

void
gs_fade_finish (GSFade *fade)
{
	g_return_if_fail (GS_IS_FADE (fade));

	if (! fade->priv->active)
	{
		return;
	}

	gs_fade_stop (fade);

	g_signal_emit (fade, signals [FADED], 0);

	fade->priv->active = FALSE;
}

static gboolean
fade_out_timer (GSFade *fade)
{
	gboolean res;

	res = gs_fade_out_iter (fade);

	/* if failed then fade is complete */
	if (! res)
	{
		gs_fade_finish (fade);
		return FALSE;
	}

	return TRUE;
}

gboolean
gs_fade_get_active (GSFade *fade)
{
	g_return_val_if_fail (GS_IS_FADE (fade), FALSE);

	return fade->priv->active;
}

static void
gs_fade_set_timeout (GSFade   *fade,
                     guint     timeout)
{
	g_return_if_fail (GS_IS_FADE (fade));

	fade->priv->timeout = timeout;
}

static void
gs_fade_start (GSFade *fade,
               guint   timeout)
{
	guint steps_per_sec = 30;
	guint msecs_per_step;
	struct GSFadeScreenPrivate *screen_priv;
	gboolean active_fade, res;
	int i;

	g_return_if_fail (GS_IS_FADE (fade));

	for (i = 0; i < fade->priv->num_screens; i++)
	{
		screen_priv = &fade->priv->screen_priv[i];
		if (screen_priv->fade_type != FADE_TYPE_NONE)
		{
			res = screen_priv->fade_setup (fade, i);
			if (res == FALSE)
				return;
		}
	}

	if (fade->priv->timer_id > 0)
	{
		gs_fade_stop (fade);
	}

	fade->priv->active = TRUE;

	gs_fade_set_timeout (fade, timeout);

	active_fade = FALSE;
	for (i = 0; i < fade->priv->num_screens; i++)
	{
		screen_priv = &fade->priv->screen_priv[i];
		if (screen_priv->fade_type != FADE_TYPE_NONE)
			active_fade = TRUE;
	}
	if (active_fade)
	{
		guint num_steps;

		num_steps = (fade->priv->timeout / 1000) * steps_per_sec;
		msecs_per_step = 1000 / steps_per_sec;
		fade->priv->alpha_per_iter = 1.0 / (gdouble)num_steps;

		fade->priv->timer_id = g_timeout_add (msecs_per_step, (GSourceFunc)fade_out_timer, fade);
	}
	else
	{
		gs_fade_finish (fade);
	}
}

typedef struct
{
	GSFadeDoneFunc done_cb;
	gpointer       data;
} FadedCallbackData;

static void
gs_fade_async_callback (GSFade            *fade,
                        FadedCallbackData *cdata)
{
	g_signal_handlers_disconnect_by_func (fade,
	                                      gs_fade_async_callback,
	                                      cdata);

	if (cdata->done_cb)
	{
		cdata->done_cb (fade, cdata->data);
	}

	g_free (cdata);
}

void
gs_fade_async (GSFade        *fade,
               guint          timeout,
               GSFadeDoneFunc func,
               gpointer       data)
{
	g_return_if_fail (GS_IS_FADE (fade));

	/* if fade is active then pause it */
	if (fade->priv->active)
	{
		gs_fade_stop (fade);
	}

	if (func)
	{
		FadedCallbackData *cb_data;

		cb_data = g_new0 (FadedCallbackData, 1);
		cb_data->done_cb = func;
		cb_data->data = data;

		g_signal_connect (fade, "faded",
		                  G_CALLBACK (gs_fade_async_callback),
		                  cb_data);
	}

	gs_fade_start (fade, timeout);
}

static void
gs_fade_sync_callback (GSFade *fade,
                       int    *flag)
{
	*flag = TRUE;
	g_signal_handlers_disconnect_by_func (fade,
	                                      gs_fade_sync_callback,
	                                      flag);
}

void
gs_fade_sync (GSFade        *fade,
              guint          timeout)
{
	int      flag = FALSE;

	g_return_if_fail (GS_IS_FADE (fade));

	/* if fade is active then pause it */
	if (fade->priv->active)
	{
		gs_fade_stop (fade);
	}

	g_signal_connect (fade, "faded",
	                  G_CALLBACK (gs_fade_sync_callback),
	                  &flag);

	gs_fade_start (fade, timeout);

	while (! flag)
	{
		gtk_main_iteration ();
	}
}

void
gs_fade_reset (GSFade *fade)
{
	int i;
	g_return_if_fail (GS_IS_FADE (fade));

	gs_debug ("Resetting fade");

	if (fade->priv->active)
	{
		gs_fade_stop (fade);
	}

	fade->priv->current_alpha = 1.0;

	gs_fade_set_alpha (fade, fade->priv->current_alpha);

	for (i = 0; i < fade->priv->num_screens; i++)
		fade->priv->screen_priv[i].fade_finish (fade, i);
}

static void
gs_fade_class_init (GSFadeClass *klass)
{
	GObjectClass   *object_class = G_OBJECT_CLASS (klass);

	object_class->finalize = gs_fade_finalize;

	signals [FADED] =
	    g_signal_new ("faded",
	                  G_TYPE_FROM_CLASS (object_class),
	                  G_SIGNAL_RUN_LAST,
	                  G_STRUCT_OFFSET (GSFadeClass, faded),
	                  NULL,
	                  NULL,
	                  g_cclosure_marshal_VOID__VOID,
	                  G_TYPE_NONE,
	                  0, G_TYPE_NONE);

	g_type_class_add_private (klass, sizeof (GSFadePrivate));
}

static void
gs_fade_init (GSFade *fade)
{
	GdkDisplay *display;
	int i;

	fade->priv = GS_FADE_GET_PRIVATE (fade);

	fade->priv->timeout = 1000;
	fade->priv->current_alpha = 1.0;

	display = gdk_display_get_default ();
	fade->priv->num_screens = gdk_display_get_n_screens (display);

	fade->priv->screen_priv = g_new0 (struct GSFadeScreenPrivate, fade->priv->num_screens);

	for (i = 0; i < fade->priv->num_screens; i++)
	{
		check_randr_extension (fade, i);
		if (!fade->priv->screen_priv[i].fade_type)
			check_gamma_extension (fade, i);
		gs_debug ("Fade type: %d", fade->priv->screen_priv[i].fade_type);
	}
}

static void
gs_fade_finalize (GObject *object)
{
	GSFade *fade;
	int i;

	g_return_if_fail (object != NULL);
	g_return_if_fail (GS_IS_FADE (object));

	fade = GS_FADE (object);

	g_return_if_fail (fade->priv != NULL);

	for (i = 0; i < fade->priv->num_screens; i++)
		fade->priv->screen_priv[i].fade_finish(fade, i);

	if (fade->priv->screen_priv)
	{
		for (i = 0; i < fade->priv->num_screens; i++)
		{
			if (!fade->priv->screen_priv[i].rrscreen)
				continue;
			mate_rr_screen_destroy (fade->priv->screen_priv[i].rrscreen);
		}

		g_free (fade->priv->screen_priv);
		fade->priv->screen_priv = NULL;
	}

	G_OBJECT_CLASS (gs_fade_parent_class)->finalize (object);
}

GSFade *
gs_fade_new (void)
{
	if (fade_object)
	{
		g_object_ref (fade_object);
	}
	else
	{
		fade_object = g_object_new (GS_TYPE_FADE, NULL);
		g_object_add_weak_pointer (fade_object,
		                           (gpointer *) &fade_object);
	}

	return GS_FADE (fade_object);
}