/*
 *  main.c
 *	part of galculator
 *  	(c) 2002-2014 Simon Flöry (simon.floery@rechenraum.com)
 *
 *  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 Library 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.
 */
 
/*
 * Initial main.c file generated by Glade. Edit as required.
 * Glade will not overwrite this file.
 */
 
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>

#include "calc_basic.h"
#include "galculator.h"
#include "display.h"
#include "config_file.h"
#include "general_functions.h"
#include "ui.h"

/* i18n */

#include <libintl.h>
#define _(String) gettext (String)
#define gettext_noop(String) String
#define N_(String) gettext_noop (String)

#include <gtk/gtk.h>
#include <gdk/gdkkeysyms.h>
#include <glib.h>
#include <glib/gstdio.h>

#define MASK_NUMLOCK GDK_MOD2_MASK

s_preferences		prefs;
s_current_status 	current_status = {0, 0, 0, 0, FALSE, FALSE, TRUE};
s_array				memory;
s_constant 			*constant;
s_user_function		*user_function;
ALG_OBJECT			*main_alg;

void print_usage ()
{
    printf (_("\n%s v%s, (c) 2002-2014 Simon Flöry\n\n\
Usage: %s [options]\n\n\
options:\n\
(GTK options)\n\
 -h, --help\t\tShow this usage message\n\
 -v, --version\t\tShow version information\n\n\
Compiled against GTK version %i.%i.%i\n\
Linked against GTK version %i.%i.%i\n\
%s-precision floating point numbers.\n"),
PACKAGE, VERSION, PACKAGE,
GTK_MAJOR_VERSION, GTK_MINOR_VERSION, GTK_MICRO_VERSION,
gtk_major_version, gtk_minor_version, gtk_micro_version,
#if USE_LIBQUADMATH
"Quad"
#else // USE_LIBQUADMATH
"Double"
#endif // USE_LIBQUADMATH
);
}

/* see GtkWidget can-activate-accel signal */

/* gtk_key_snooper_install is deprecated from version 3.4 on. Anyway, the key
 * snooper was a dirty work-around for problems I had with early versions of
 * GTK2, that I can't reproduce nowadays. So try to go w/o any key snopper for
 * GTK3.
 */
#if !(GTK_CHECK_VERSION(3, 4, 0))

int key_snooper (GtkWidget *grab_widget, GdkEventKey *event, gpointer func_data)
{
	GtkWidget	*formula_entry;
	
	/* the problem: key acceleration in gtk2 works a bit strange. I do not
	 * understand it completely. The following is in part the result of a
	 * long trial and error process. If you can explain why it works, please
	 * tell me.
	 * btw, do this only if main_window is the current window. otherwise, 
	 * keypad's 0,2,4,6,8 won't work in gtkentry etc. (e.g. found in prefs)
	 */
	
	//fprintf (stderr, "[%s] key snooper (1): %i %i %s\n", PROG_NAME, event->state, event->keyval, gdk_keyval_name (event->keyval));
#if 0	
	if (((event->keyval != GDK_KP_2) && (event->keyval != GDK_KP_Down) &&
		(event->keyval != GDK_KP_4) && (event->keyval != GDK_KP_Left) &&
		(event->keyval != GDK_KP_6) && (event->keyval != GDK_KP_Right) &&
		(event->keyval != GDK_KP_8) && (event->keyval != GDK_KP_Up) &&
		(event->keyval != GDK_KP_0) && (event->keyval != GDK_KP_Insert)) ||
		(strcmp (gtk_buildable_get_name (GTK_BUILDABLE(gtk_widget_get_toplevel(grab_widget))), "main_window") != 0) ||
		(prefs.mode == PAPER_MODE))
			event->state &= ~GDK_MOD2_MASK;
#endif
	//fprintf (stderr, "[%s] key snooper (2): %i %i %s\n", PROG_NAME, event->state, event->keyval, gdk_keyval_name (event->keyval));
	
	/* another problem: we have keyboard accelerators which are simple
	 * keypresses, e.g. "1", "2" but also "s" for the sin button. if the
	 * formula entry is active and user types a "s", user expects a s to
	 * be appended. moreover the same for special keys like backspace etc.
	 * unfortunately, accelerators have higher priority and can't be somehow
	 * blocked (they can? tell me!). therefore if formula_entry is active
	 * and it's a "simple" key press (is this the best solution?), then emit
	 * the signal directly and return TRUE to prevent any further 
	 * procession.
	 */
	
	if (((formula_entry = formula_entry_is_active(grab_widget)) != NULL) && 
		(event->type == GDK_KEY_PRESS)) {
		if ((event->state == 0) || (event->state == GDK_SHIFT_MASK)) {
			gtk_widget_event (formula_entry, (GdkEvent *)event);
			return TRUE;
		}
	}
	
	return FALSE;
}
#endif

/*
 * main
 */

#ifdef WITH_HILDON
/* Registering the hildon application */
static GtkWidget* glade_hildon_window_new (GtkBuilder *xml, GType type, GladeWidgetInfo *info)
{
    return hildon_window_new();
}
#endif

int main (int argc, char *argv[])
{
    char		*config_file_name;
	GtkWidget 	*main_window;

#ifdef WITH_HILDON
	HildonProgram   *hildon_program;
#endif
	
	/*
	 * gtk_init runs (among other things) setlocale (LC_ALL, ""). Therefore we
	 * have to/can deal with i18n only from this point.
	 * call gtk_init before getopts to get gtk options stripped (--display etc)
	 */

	gtk_init (&argc, &argv);
	
	bindtextdomain (PACKAGE, PACKAGE_LOCALE_DIR);
	bind_textdomain_codeset (PACKAGE, "UTF-8");
	textdomain (PACKAGE);
    	
	if (argc > 1) {
		print_usage ();
		return EXIT_SUCCESS;
	}

    /* get config file. 
     * 
     * at first we check if there is env variable GALCULATOR_CONFIG set. If not,
     * we try the XDG location with g_get_user_config_dir. If this does not 
     * succeed, we try the old non-spec location, that will be migrated to the
     * XDG location.
     */
    config_file_name = getenv("GALCULATOR_CONFIG");
    if (config_file_name) config_file_name = g_strdup(config_file_name);
    else {
		/* if there is a config file in g_get_user_config_dir, use that. 
		 * otherwise, try old method. anyway, we will store in 
		 * g_get_user_config_dir and tell the user to remove the file from the
		 * old location.
		 */
		config_file_name = g_strdup_printf("%s/%s/%s", g_get_user_config_dir(), PACKAGE, CONFIG_FILE_NAME);
		if (!g_file_test(config_file_name, G_FILE_TEST_IS_REGULAR)) {
			/* create directory for new config file location */
			char *path = g_strdup_printf("%s/%s", g_get_user_config_dir(), PACKAGE);
			if (g_mkdir(path, 0755) == -1 && errno != EEXIST) 
				fprintf (stderr, _("[%s] configuration file: Failed to create directory %s.\n"), PACKAGE, path);
			g_free(path);
			/* check old file location */
			char *config_file_name_old = g_strdup_printf("%s/%s", getenv ("HOME"), CONFIG_FILE_NAME_OLD);
			if (g_file_test(config_file_name_old, G_FILE_TEST_IS_REGULAR)) {
				fprintf (stderr, _("[%s] configuration file: We will move the configuration \
file from %s to %s. After you quit %s, you may remove the configuration file from its old location %s.\n"), 
PACKAGE, config_file_name_old, config_file_name, PACKAGE, config_file_name_old);
				g_free(config_file_name);
				config_file_name = config_file_name_old;
			} else g_free(config_file_name_old);
		}
	}

	prefs = config_file_read (config_file_name);
	constant = config_file_get_constants();
    user_function = config_file_get_user_functions();

	current_status.notation = prefs.def_notation;

#ifdef WITH_HILDON
	glade_register_widget (HILDON_TYPE_WINDOW, glade_hildon_window_new, glade_standard_build_children, NULL);
	hildon_program = HILDON_PROGRAM(hildon_program_get_instance());
#endif

	/* at first get the main frame */
	
	/* sth like ui_launch_up_ui for splitting into first time wizard? */
	main_window = ui_main_window_create();
#ifdef WITH_HILDON
	hildon_program_add_window(hildon_program, HILDON_WINDOW(main_window));
	g_set_application_name("Galculator");
	create_hildon_menu(HILDON_WINDOW(main_window));
#endif	
	gtk_window_set_title ((GtkWindow *)main_window, PACKAGE);

	/* usually, only Shift, CTRL and ALT modifiers are paid attention to by 
	 * accelerator code. add MOD2 (NUMLOCK allover the world?) to the list. 
	 * We have to do this for a working keypad.
	 */

	gtk_accelerator_set_default_mod_mask (gtk_accelerator_get_default_mod_mask () | GDK_MOD2_MASK); 
				  
	/* prepare calc_basic */

	main_alg = alg_init (0);
	rpn_init (prefs.stack_size, 0);
		
	/* apply changes */
	apply_preferences (prefs);
    remember_display_values ();

	memory.data = NULL;
	memory.len = 0;

	/* see function key_snooper for details */
#if !(GTK_CHECK_VERSION(3, 0, 0))
    gtk_key_snooper_install (key_snooper, NULL);
#endif
	
    /* make the window as small as possible */
	gtk_window_resize ((GtkWindow *)main_window, 1, 1);

	/* gtk_widget_show main window as late as possible */
	gtk_widget_show (main_window);
	
	/*
	 * enter GTK's main loop
	 */
	
	gtk_main ();

	/* 
	 * save changes to configuration file 
	 */

	if (!getenv("GALCULATOR_CONFIG")) {
		/* If the config file name is not from an env variable, enforce the
		 * new location.
		 */
		g_free(config_file_name);
		config_file_name = g_strdup_printf ("%s/%s/%s", g_get_user_config_dir(), PACKAGE, CONFIG_FILE_NAME);
	}
	config_file_write (config_file_name, prefs, constant, user_function);
	g_free (config_file_name);

	return EXIT_SUCCESS;
}
