/* GTK - The GIMP Toolkit
 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Library 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
 * Library General Public License for more details.
 *
 * You should have received a copy of the GNU Library 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 <stdlib.h>
#include <gtk/gtk.h>
#include <time.h>

/* Backing pixmap for drawing area */
// Erre a változóra azért van szükség, hogy elmentsük az eddig rajzolt képet,
// hogy pl. áthelyezéskor, tálcáról visszahozáskor megmaradjon a kép
static GdkPixmap *pixmap = NULL;

// Itt hozhatunk létre globális változókat.
// Ezek kezdeti értékeit a main elején érdemes beállítani.
// A globális változók végig a program futása során elérhetőek bármely függvényből.
int m_counter = 0;
int m_timer = 0;
int m_quick = 0;
int m_width = 0;
int m_height = 0;
int m_x = 50;
int m_y = 50;


/* Create a new backing pixmap of the appropriate size */
// Induláskor, illetve átméretezéskor hívódik meg
// Feladata, hogy létrehozza (ha eddig is létre volt hozva, akkor törölje)
// és fehér színnel töltse meg a pixmap változót
static gboolean configure_event( GtkWidget         *widget,
                                 GdkEventConfigure *event )
{
  if (pixmap)
    g_object_unref (pixmap);

  pixmap = gdk_pixmap_new (widget->window,
			   widget->allocation.width,
			   
			   widget->allocation.height,
			   -1);
  gdk_draw_rectangle (pixmap,
		      widget->style->white_gc,
		      TRUE,
		      0, 0,
		      widget->allocation.width,
		      widget->allocation.height);

  m_width = widget->allocation.width;
  m_height = widget->allocation.height;

  return TRUE;
}

/* Redraw the screen from the backing pixmap */
// Mozgatás, tálcáról visszaméretezés során kirajzolja a pixmap-ből
// az elmentett képet
// Egér kattintás során a draw_brush függvényben az utolsó sor
// a gtk_widget_queue_draw_area függvény hatására is innen frissítődik
// a kép
static gboolean expose_event( GtkWidget      *widget,
                              GdkEventExpose *event )
{
  gdk_draw_drawable (widget->window,
		     widget->style->fg_gc[GTK_WIDGET_STATE (widget)],
		     pixmap,
		     event->area.x, event->area.y,
		     event->area.x, event->area.y,
		     event->area.width, event->area.height);

  return FALSE;
}


/* Draw a rectangle on the screen */
// Egér kattintásakor, illetve mozgatásakor fogjuk ezt a függvényt meghívni
// Célja, hogy rajzoljon egy téglalapot a megadott színnel
static void draw_brush( GtkWidget *widget,
                        gdouble    x,
                        gdouble    y,
			GdkGC*	gc)
{
// Létrehozunk egy update_rect nevű változót, amely GdkRectangle nevű
// objectum típusú lesz. A lényeg: van update_rect.x, update_rect.y,
// update_rect.width és update_rect.height egész számunk, amelyeket
// kedvünkre változtathatunk.
  GdkRectangle update_rect;

// A draw_brush paramétereiben megkapjuk x és y értékét. Ezeket átkonvertáljuk
// egésszé a (gintx) segítségével. Ezt nem kellene megcsinálnunk, megtörténne 
// automatikusan, de így nem kapunk róla figyelmeztetést
//
// x-5 és y-5 koordinátáknál kezdődik a téglalap, és 10x10-es méretű
  update_rect.x = (gint)x - 5;
  update_rect.y = (gint)y - 5;
  update_rect.width = 10;
  update_rect.height = 10;
// Berajzoljuk a pixmap-ba gc -ben leírt paraméterrel
// A változás lényegében a szín lesz
  gdk_draw_rectangle (pixmap,
		      gc,
		      TRUE,
		      update_rect.x, update_rect.y,
		      update_rect.width, update_rect.height);
  // A kövektező sor hatására frissíti a téglalap részt.
  // Emiatt meghívódik az expose_event, ami pedig
  // a pixmap-ből kimásolja a megfelelő részt.
  gtk_widget_queue_draw_area (widget, 
		              update_rect.x, update_rect.y,
		              update_rect.width, update_rect.height);
}

/* Draw a rectangle on the screen */
// Egér kattintásakor, illetve mozgatásakor fogjuk ezt a függvényt meghívni
// Célja, hogy rajzoljon egy téglalapot a piros,zöld,kék komponensekkel megadott
// színnel (az egyes komponensek értékei 0 és 65535 közöttiek lehetnek)
static void draw_brush( GtkWidget *widget,
                        gdouble    x,
                        gdouble    y,
			guint16 red,
			guint16 green,
			guint16 blue)
{
	GdkColor color;
	color.red = red;
	color.green = green;
	color.blue = blue;
	gdk_gc_set_rgb_fg_color( widget->style->mid_gc[GTK_WIDGET_STATE (widget)], &color);
	draw_brush (widget, x, y,widget->style->mid_gc[GTK_WIDGET_STATE (widget)]);
}
static gboolean draw_button_event( GtkWidget      *widget,
                              GdkEventExpose *event )
{
// Részletes leírás: http://developer.gnome.org/doc/API/2.2/gtk/GtkMessageDialog.html
 GtkWidget* dialog = gtk_message_dialog_new (NULL,
                                  GTK_DIALOG_DESTROY_WITH_PARENT,
                                  GTK_MESSAGE_QUESTION,
                                  GTK_BUTTONS_YES_NO,
                                  "Lenyomtad a gombot!!!\nRakjunk le sok kis 4zetet?");
 gint ret = gtk_dialog_run (GTK_DIALOG (dialog));
 gtk_widget_destroy (dialog);

 if (ret == GTK_RESPONSE_YES)
 {
	 for (int i = 0; i < m_width/20; i++)
	 {
		 for (int j = 0; j < m_height/10; j++)
		 {
			 draw_brush(widget, i*20+m_counter,j*20,widget->style->black_gc);
		 }		 
	 }
	 m_counter++;
 }
 return  FALSE;
}

static gboolean button_press_event( GtkWidget      *widget,
                                    GdkEventButton *event )
{
		   
  if (event->button == 1 && pixmap != NULL)
    draw_brush (widget, event->x, event->y, widget->style->black_gc);
 
  if (event->button == 3 && pixmap != NULL)
    draw_brush (widget, event->x, event->y, widget->style->white_gc);

  return TRUE;
}

static gboolean key_press_event( GtkWidget      *widget,
                                    GdkEventKey *event, gpointer data )
{
  if (event->keyval == 65361 && pixmap != NULL)
  {
	draw_brush (widget, m_x, m_y, widget->style->white_gc);
	m_x -= 10;
	draw_brush (widget, m_x, m_y, widget->style->black_gc);
  }
  if (event->keyval == 65362 && pixmap != NULL)
  {
	draw_brush (widget, m_x, m_y, widget->style->white_gc);
	m_y -= 10;
	draw_brush (widget, m_x, m_y, widget->style->black_gc);
  }
  if (event->keyval == 65363 && pixmap != NULL)
  {
	draw_brush (widget, m_x, m_y, widget->style->white_gc);
	m_x += 10;
	draw_brush (widget, m_x, m_y, widget->style->black_gc);
  }
  if (event->keyval == 65364 && pixmap != NULL)
  {
	draw_brush (widget, m_x, m_y, widget->style->white_gc);
	m_y += 10;
	draw_brush (widget, m_x, m_y, widget->style->black_gc);
  }
  return TRUE;
}
// Egér mozgatásakor hívódik meg
static gboolean motion_notify_event( GtkWidget *widget,
                                     GdkEventMotion *event )
{
  int x, y;
  GdkModifierType state;

  if (event->is_hint)
    gdk_window_get_pointer (event->window, &x, &y, &state);
  else
    {
      x = (int)event->x;
      y = (int)event->y;
      state = (GdkModifierType) event->state;
    }
  
  // A & jelentése: bitenkénti ÉS
  // Ez elsőre fura lehet. A state egy változó, aminek az értéke egy szám
  // A számot a gép biteken tárolja, pl. 10011011
  // A GDK_BUTTON1_MASK egy olyan értéket takar, amelyik kettő hatvány,
  // így biteken ábrázolva 00010000 alakú. A két szám bitenkétni ÉS művelete
  // az a szám, ami bitenként ábrázolva ott tartalmaz 1-est, ahol mind a kettő
  // 1-est tartalmazott. Az if olyan, hogy ha egy szám nem nulla, akkor az
  // igazat jelent. A && már sima ÉS kapcsolat.
  if (state & GDK_BUTTON1_MASK && pixmap != NULL)
    draw_brush (widget, x, y,     m_timer,m_quick,m_quick);

  if (state & GDK_BUTTON3_MASK && pixmap != NULL)
    draw_brush (widget, x, y, widget->style->white_gc);
  
  return TRUE;
}

void quit ()
{
  exit (0);
}

//Ez a függvény másodpercenként lefut egyszer
gint my_timer(GtkWidget* window)
{
	g_print("A my_timer fuggveny lefutott.\n");
	int i = rand()%m_width;
	int j = rand()%m_height;

	draw_brush (window, i, j,  65000,65000,m_timer);

	m_timer = (m_timer + 10000)%65536;
	
//Ha 1-gyel térünk vissza, akkor meghívódik mégegyszer.
	return(1); 
}

//Ez a függvény másodpercenként 20-szor fut le
gint my_timer_quick(gpointer * p)
{
	m_quick = (m_quick + 5000)%65536;
	
//Ha 1-gyel térünk vissza, akkor meghívódik mégegyszer.
	return(1); 
}
int main( int   argc, 
          char *argv[] )
{
 //Változók kezdeti beállítását elvégezhetjük itt
  srand(time(NULL));

  GtkWidget *window;
  GtkWidget *drawing_area;
  GtkWidget *vbox;

  GtkWidget *draw_button;
  GtkWidget *quit_button;

  gtk_init (&argc, &argv);

  window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
  gtk_widget_set_name (window, "Test Input");

  vbox = gtk_vbox_new (FALSE, 0);
  gtk_container_add (GTK_CONTAINER (window), vbox);
  gtk_widget_show (vbox);

  g_signal_connect (G_OBJECT (window), "destroy",
                    G_CALLBACK (quit), NULL);

  /* Create the drawing area */

  drawing_area = gtk_drawing_area_new ();
  gtk_widget_set_size_request (GTK_WIDGET (drawing_area), 200, 200);
  gtk_box_pack_start (GTK_BOX (vbox), drawing_area, TRUE, TRUE, 0);

  gtk_widget_show (drawing_area);

  /* Signals used to handle backing pixmap */

  g_signal_connect (G_OBJECT (drawing_area), "expose_event",
		    G_CALLBACK (expose_event), NULL);
  g_signal_connect (G_OBJECT (drawing_area),"configure_event",
		    G_CALLBACK (configure_event), NULL);

  /* Event signals */

  g_signal_connect (G_OBJECT (drawing_area), "motion_notify_event",
		    G_CALLBACK (motion_notify_event), NULL);
  g_signal_connect (G_OBJECT (drawing_area), "button_press_event",
		    G_CALLBACK (button_press_event), NULL);
  g_signal_connect (G_OBJECT (window), "key_press_event",
		    G_CALLBACK (key_press_event), NULL);

  gtk_widget_add_events(window, GDK_KEY_PRESS_MASK);

  gtk_widget_set_events (drawing_area, GDK_EXPOSURE_MASK
			 | GDK_LEAVE_NOTIFY_MASK
			 | GDK_BUTTON_PRESS_MASK
//			 | GDK_KEY_PRESS_MASK
			 | GDK_POINTER_MOTION_MASK
			 | GDK_POINTER_MOTION_HINT_MASK);

  /* .. And a quit button */
  // Draw gomb:
  draw_button = gtk_button_new_with_label ("Draw");
  gtk_box_pack_start (GTK_BOX (vbox), draw_button, FALSE, FALSE, 0);

  g_signal_connect (G_OBJECT (draw_button), "clicked",
		    G_CALLBACK (draw_button_event), NULL);
  gtk_widget_show (draw_button);


  //Quit gomb:
  quit_button = gtk_button_new_with_label ("Quit");
  gtk_box_pack_start (GTK_BOX (vbox), quit_button, FALSE, FALSE, 0);

  g_signal_connect_swapped (G_OBJECT (quit_button), "clicked",
			    G_CALLBACK (gtk_widget_destroy),
			    G_OBJECT (window));
  gtk_widget_show (quit_button);
  
  gtk_widget_show (window);

  //időzítő
  int htimeout;
  htimeout = gtk_timeout_add((guint32)1000,(GtkFunction)my_timer,window);
  int htimeout2;
  htimeout2 = gtk_timeout_add((guint32)50,(GtkFunction)my_timer_quick,window);

  gtk_main ();

  gtk_timeout_remove(htimeout);

  return 0;
}

