/*
 * Copyright © 2011 Intel Corporation
 *
 * Permission to use, copy, modify, distribute, and sell this software and its
 * documentation for any purpose is hereby granted without fee, provided that
 * the above copyright notice appear in all copies and that both that copyright
 * notice and this permission notice appear in supporting documentation, and
 * that the name of the copyright holders not be used in advertising or
 * publicity pertaining to distribution of the software without specific,
 * written prior permission.  The copyright holders make no representations
 * about the suitability of this software for any purpose.  It is provided "as
 * is" without express or implied warranty.
 *
 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
 * OF THIS SOFTWARE.
 */

#include <stdlib.h>
#include <stdio.h>
#include <linux/input.h>

#include "compositor.h"

struct wlsc_switcher {
	struct wlsc_compositor *compositor;
	struct wlsc_surface *current;
	struct wl_listener listener;
};

static void
wlsc_switcher_next(struct wlsc_switcher *switcher)
{
	struct wl_list *l;
	struct wl_surface *current;

	wlsc_surface_damage(switcher->current);
	l = switcher->current->link.next;
	if (l == &switcher->compositor->surface_list)
		l = switcher->compositor->surface_list.next;
	switcher->current = container_of(l, struct wlsc_surface, link);
	wl_list_remove(&switcher->listener.link);
	current = &switcher->current->surface;
	wl_list_insert(current->resource.destroy_listener_list.prev,
		       &switcher->listener.link);
	switcher->compositor->overlay = switcher->current;
	wlsc_surface_damage(switcher->current);
}

static void
switcher_handle_surface_destroy(struct wl_listener *listener,
				struct wl_resource *resource, uint32_t time)
{
	struct wlsc_switcher *switcher =
		container_of(listener, struct wlsc_switcher, listener);

	wlsc_switcher_next(switcher);
}

static struct wlsc_switcher *
wlsc_switcher_create(struct wlsc_compositor *compositor)
{
	struct wlsc_switcher *switcher;

	switcher = malloc(sizeof *switcher);
	switcher->compositor = compositor;
	switcher->current = container_of(compositor->surface_list.next,
					 struct wlsc_surface, link);
	switcher->listener.func = switcher_handle_surface_destroy;
	wl_list_init(&switcher->listener.link);

	return switcher;
}

static void
wlsc_switcher_destroy(struct wlsc_switcher *switcher)
{
	wl_list_remove(&switcher->listener.link);
	free(switcher);
}

static void
switcher_next_binding(struct wl_input_device *device, uint32_t time,
		      uint32_t key, uint32_t button,
		      uint32_t state, void *data)
{
	struct wlsc_compositor *compositor = data;

	if (!state)
		return;
	if (wl_list_empty(&compositor->surface_list))
		return;
	if (compositor->switcher == NULL)
		compositor->switcher = wlsc_switcher_create(compositor);

	wlsc_switcher_next(compositor->switcher);
}

static void
switcher_terminate_binding(struct wl_input_device *device,
			   uint32_t time, uint32_t key, uint32_t button,
			   uint32_t state, void *data)
{
	struct wlsc_compositor *compositor = data;
	struct wlsc_input_device *wd = (struct wlsc_input_device *) device;

	if (compositor->switcher && !state) {
		wlsc_surface_activate(compositor->switcher->current, wd, time);
		wlsc_switcher_destroy(compositor->switcher);
		compositor->switcher = NULL;
		compositor->overlay = NULL;
	}
}

void
wlsc_switcher_init(struct wlsc_compositor *compositor)
{
	wlsc_compositor_add_binding(compositor,
				    KEY_TAB, 0, MODIFIER_SUPER,
				    switcher_next_binding, compositor);
	wlsc_compositor_add_binding(compositor,
				    KEY_LEFTMETA, 0, MODIFIER_SUPER,
				    switcher_terminate_binding, compositor);
	wlsc_compositor_add_binding(compositor,
				    KEY_RIGHTMETA, 0, MODIFIER_SUPER,
				    switcher_terminate_binding, compositor);
}