/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
/* e-storage-browser.c
 *
 * Copyright (C) 2003  Ximian, Inc.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * 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., 59 Temple Place - Suite 330,
 * Boston, MA 02111-1307, USA.
 *
 * Author: Ettore Perazzoli <ettore@ximian.com>
 */

/* TODO:

  - Currently it assumes that the starting path always exists, and you
    can't remove it.  It might be a limitation, but it makes the logic
    very simple and robust.

   - Doesn't save expansion state for nodes.

   - Context menu handling?

*/

#include <config.h>

#include "e-storage-browser.h"

#include "e-shell-marshal.h"
#include "e-storage-set-view.h"

#include <gal/util/e-util.h>

#include <gtk/gtknotebook.h>
#include <gtk/gtkscrolledwindow.h>
#include <string.h>


#define PARENT_TYPE G_TYPE_OBJECT
static GObjectClass *parent_class = NULL;


struct _EStorageBrowserPrivate {
	char *starting_path;
	char *current_path;

	GtkWidget *view_notebook;
	GtkWidget *storage_set_view;
	GtkWidget *storage_set_view_scrolled;

	GHashTable *path_to_view; /* (char *, GtkWidget *) */

	EStorageBrowserCreateViewCallback create_view_callback;
	void *create_view_callback_data;
};


enum {
	WIDGETS_GONE,
	PAGE_SWITCHED,
	NUM_SIGNALS
};

static unsigned int signals[NUM_SIGNALS] = { 0 };


/* Callbacks.  */

static void
storage_set_view_folder_selected_callback (EStorageSetView *storage_set_view,
					   const char *path,
					   EStorageBrowser *browser)
{
	if (! e_storage_browser_show_path (browser, path)) {
		/* Make the selection go back to where it was.  */
		e_storage_browser_show_path (browser, browser->priv->current_path);
	}
}

static void
storage_set_removed_folder_callback (EStorageSet *storage_set,
				     const char *path,
				     EStorageBrowser *browser)
{
	if (g_hash_table_lookup (browser->priv->path_to_view, path) != NULL)
		e_storage_browser_remove_view_for_path (browser, path);
}

static void
view_notebook_weak_notify (EStorageBrowser *browser)
{
	browser->priv->view_notebook = NULL;

	if (browser->priv->storage_set_view == NULL)
		g_signal_emit (browser, signals[WIDGETS_GONE], 0);
}

static void
storage_set_view_weak_notify (EStorageBrowser *browser)
{
	browser->priv->storage_set_view = NULL;

	if (browser->priv->view_notebook == NULL)
		g_signal_emit (browser, signals[WIDGETS_GONE], 0);
}


/* GObject methods.  */

static void
impl_dispose (GObject *object)
{
	EStorageBrowserPrivate *priv = E_STORAGE_BROWSER (object)->priv;

	if (priv->view_notebook != NULL) {
		g_object_weak_unref (G_OBJECT (priv->view_notebook),
				     (GWeakNotify) view_notebook_weak_notify,
				     object);
		priv->view_notebook = NULL;
	}

	if (priv->storage_set_view != NULL) {
		g_object_weak_unref (G_OBJECT (priv->storage_set_view),
				     (GWeakNotify) storage_set_view_weak_notify,
				     object);
		priv->storage_set_view = NULL;
	}

	(* G_OBJECT_CLASS (parent_class)->dispose) (object);
}

static void
impl_finalize (GObject *object)
{
	EStorageBrowserPrivate *priv = E_STORAGE_BROWSER (object)->priv;

	g_free (priv->starting_path);
	g_free (priv->current_path);

	g_hash_table_destroy (priv->path_to_view);

	g_free (priv);

	(* G_OBJECT_CLASS (parent_class)->finalize) (object);
}


/* Initialization.  */

static void
class_init (EStorageBrowserClass *class)
{
	GObjectClass *object_class = G_OBJECT_CLASS (class);

	object_class->dispose  = impl_dispose;
	object_class->finalize = impl_finalize;

	parent_class = g_type_class_peek_parent (class);

	signals[WIDGETS_GONE]
		= g_signal_new ("widgets_gone",
				G_OBJECT_CLASS_TYPE (object_class),
				G_SIGNAL_RUN_LAST,
				G_STRUCT_OFFSET (EStorageBrowserClass, widgets_gone),
				NULL, NULL,
				e_shell_marshal_NONE__NONE,
				G_TYPE_NONE, 0);

	signals[PAGE_SWITCHED]
		= g_signal_new ("page_switched",
				G_OBJECT_CLASS_TYPE (object_class),
				G_SIGNAL_RUN_LAST,
				G_STRUCT_OFFSET (EStorageBrowserClass, page_switched),
				NULL, NULL,
				e_shell_marshal_NONE__POINTER_POINTER,
				G_TYPE_NONE, 2,
				G_TYPE_POINTER, G_TYPE_POINTER);
}

static void
init (EStorageBrowser *browser)
{
	EStorageBrowserPrivate *priv;

	priv = g_new0 (EStorageBrowserPrivate, 1);

	priv->path_to_view = g_hash_table_new_full (g_str_hash, g_str_equal,
						    (GDestroyNotify) g_free,
						    (GDestroyNotify) g_object_unref);

	priv->view_notebook = gtk_notebook_new ();
	g_object_weak_ref (G_OBJECT (priv->view_notebook), (GWeakNotify) view_notebook_weak_notify, browser);

	gtk_notebook_set_show_border (GTK_NOTEBOOK (priv->view_notebook), FALSE);
	gtk_notebook_set_show_tabs (GTK_NOTEBOOK (priv->view_notebook), FALSE);

	browser->priv = priv;
}


EStorageBrowser *
e_storage_browser_new  (EStorageSet *storage_set,
			const char *starting_path,
			EStorageBrowserCreateViewCallback create_view_callback,
			void *callback_data)
{
	EStorageBrowser *new;

	g_return_val_if_fail (create_view_callback != NULL, NULL);

	new = g_object_new (e_storage_browser_get_type (), NULL);

	new->priv->create_view_callback      = create_view_callback;
	new->priv->create_view_callback_data = callback_data;
	new->priv->starting_path             = g_strdup (starting_path);

	new->priv->storage_set_view          = e_storage_set_create_new_view (storage_set, NULL);
	gtk_widget_show (new->priv->storage_set_view);

	new->priv->storage_set_view_scrolled = gtk_scrolled_window_new (NULL, NULL);
	gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (new->priv->storage_set_view_scrolled), GTK_SHADOW_IN);
	gtk_container_add (GTK_CONTAINER (new->priv->storage_set_view_scrolled), new->priv->storage_set_view);
	gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (new->priv->storage_set_view_scrolled),
					GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);

	g_object_weak_ref (G_OBJECT (new->priv->storage_set_view), (GWeakNotify) storage_set_view_weak_notify, new);

	g_signal_connect_object (new->priv->storage_set_view,
				 "folder_selected", G_CALLBACK (storage_set_view_folder_selected_callback),
				 G_OBJECT (new), 0);
	g_signal_connect_object (e_storage_set_view_get_storage_set (E_STORAGE_SET_VIEW (new->priv->storage_set_view)),
				 "removed_folder", G_CALLBACK (storage_set_removed_folder_callback),
				 G_OBJECT (new), 0);

	if (! e_storage_browser_show_path (new, starting_path)) {
		g_object_unref (new);
		return NULL;
	}

	return new;
}


GtkWidget *
e_storage_browser_peek_tree_widget (EStorageBrowser *browser)
{
	return browser->priv->storage_set_view;
}

GtkWidget *
e_storage_browser_peek_tree_widget_scrolled (EStorageBrowser *browser)
{
	return browser->priv->storage_set_view_scrolled;
}

GtkWidget *
e_storage_browser_peek_view_widget (EStorageBrowser *browser)
{
	return browser->priv->view_notebook;
}

EStorageSet *
e_storage_browser_peek_storage_set (EStorageBrowser *browser)
{
	return e_storage_set_view_get_storage_set (E_STORAGE_SET_VIEW (browser->priv->storage_set_view));
}

gboolean
e_storage_browser_show_path  (EStorageBrowser *browser,
			      const char *path)
{
	EStorageBrowserPrivate *priv = browser->priv;
	GtkWidget *current_view;
	GtkWidget *existing_view;
	GtkWidget *new_view;
	GtkNotebook *notebook;

	notebook = GTK_NOTEBOOK (priv->view_notebook);

	current_view = gtk_notebook_get_nth_page (GTK_NOTEBOOK (priv->view_notebook),
						  gtk_notebook_get_current_page (GTK_NOTEBOOK (priv->view_notebook)));

	existing_view = g_hash_table_lookup (priv->path_to_view, path);
	if (existing_view != NULL) {
		gtk_notebook_set_current_page (notebook, gtk_notebook_page_num (notebook, existing_view));
		g_print ("page switched\n");
		g_signal_emit (browser, signals[PAGE_SWITCHED], 0, current_view, existing_view);
		return TRUE;
	}

	new_view = (* priv->create_view_callback) (browser, path, priv->create_view_callback_data);
	if (new_view == NULL)
		return FALSE;

	gtk_widget_show (new_view);
	gtk_notebook_append_page (notebook, new_view, NULL);
	gtk_notebook_set_current_page (notebook, gtk_notebook_page_num (notebook, new_view));

	g_print ("page switched\n");
	g_signal_emit (browser, signals[PAGE_SWITCHED], 0, current_view, new_view);

	g_object_ref(new_view);
	g_hash_table_insert (priv->path_to_view, g_strdup (path), new_view);

	g_free (priv->current_path);
	priv->current_path = g_strdup (path);

	e_storage_set_view_set_current_folder (E_STORAGE_SET_VIEW (priv->storage_set_view), path);

	return TRUE;
}

void
e_storage_browser_remove_view_for_path (EStorageBrowser *browser,
					const char *path)
{
	GtkWidget *view;

	if (strcmp (path, browser->priv->starting_path) == 0) {
		g_warning (G_GNUC_FUNCTION ": cannot remove starting view");
		return;
	}

	view = g_hash_table_lookup (browser->priv->path_to_view, path);
	if (view == NULL) {
		g_warning (G_GNUC_FUNCTION ": no view for %s", path);
		return;
	}

	g_hash_table_remove (browser->priv->path_to_view, path);
	gtk_widget_destroy (view);

	e_storage_browser_show_path (browser, browser->priv->starting_path);
}


E_MAKE_TYPE (e_storage_browser, "EStorageBrowser", EStorageBrowser, class_init, init, PARENT_TYPE)