Re: reparenting widgets



>Any clues?  The floating/main reparent operation is using the
>Gtk::Window::reparent and not dealing with GladeXML::reparent function. 
>Could that be a problem?
>
>Any other clues?

i got my ideas from sodipodi ...

/*
    Copyright (C) 2003 Paul Davis
 
    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., 675 Mass Ave, Cambridge, MA 02139, USA.

    $Id: tearoff.h,v 1.3 2003/10/09 01:35:21 trutkin Exp $
*/

#ifndef __gtkmmext_tearoff_h__
#define __gtkmmext_tearoff_h__

#include <gtk--.h>

namespace Gtkmmext {

class TearOff : public Gtk::HBox
{
  public:
	TearOff (Gtk::Widget& contents);
	virtual ~TearOff ();

	SigC::Signal0<void> Detach;
	SigC::Signal0<void> Attach;

	Gtk::Window* tearoff_window() const { return own_window; }
	bool torn_off() const;

  private:
	Gtk::Widget&   contents;
	Gtk::Window*   own_window;
	Gtk::Arrow     tearoff_arrow;
	Gtk::Arrow     close_arrow;
	Gtk::HBox      window_box;
	Gtk::EventBox  tearoff_event_box;
	Gtk::EventBox  close_event_box;
	double         drag_x;
	double         drag_y;
	bool           dragging;

	gint tearoff_click (GdkEventButton*);
	gint close_click (GdkEventButton*);

	gint window_motion (GdkEventMotion*);
	gint window_button_press (GdkEventButton*);
	gint window_button_release (GdkEventButton*);
	gint window_delete_event (GdkEventAny*);
};

} /* namespace */

#endif  // __gtkmmext_tearoff_h__


/*
    Copyright (C) 2003 Paul Davis
 
    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., 675 Mass Ave, Cambridge, MA 02139, USA.

    $Id: tearoff.cc,v 1.6 2003/10/22 19:54:25 pbd Exp $
*/

#include <cmath>
#include <gtkmmext/tearoff.h>

using namespace Gtkmmext;
using namespace Gtk;
using namespace std;

TearOff::TearOff (Gtk::Widget& c)
	: contents (c),
	  tearoff_arrow (GTK_ARROW_DOWN, GTK_SHADOW_OUT),
	  close_arrow (GTK_ARROW_UP, GTK_SHADOW_OUT)
{
	dragging = false;

	tearoff_event_box.add (tearoff_arrow);
	tearoff_event_box.set_events (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK);
	tearoff_event_box.button_release_event.connect (slot (*this, &TearOff::tearoff_click));

	close_event_box.add (close_arrow);
	close_event_box.set_events (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK);
	close_event_box.button_release_event.connect (slot (*this, &TearOff::close_click));
	
	own_window = new Gtk::Window (GTK_WINDOW_TOPLEVEL);
	own_window->add_events (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK|GDK_POINTER_MOTION_MASK|GDK_POINTER_MOTION_HINT_MASK);
	own_window->set_policy (false, false, false);
	own_window->realize ();
	own_window->get_window().set_decorations (GdkWMDecoration (GDK_DECOR_BORDER|GDK_DECOR_RESIZEH));
	
	VBox* box1;
	box1 = manage (new VBox);
	box1->pack_start (close_event_box, false, false, 5);
	
	window_box.pack_end (*box1, false, false, 2);
	own_window->add (window_box);
	
	own_window->button_press_event.connect (slot (*this, &TearOff::window_button_press));
	own_window->button_release_event.connect (slot (*this, &TearOff::window_button_release));
	own_window->motion_notify_event.connect (slot (*this, &TearOff::window_motion));
	own_window->delete_event.connect (slot (*this, &TearOff::window_delete_event));
	
	tearoff_arrow.set_name ("TearOffArrow");
	close_arrow.set_name ("TearOffArrow");

	VBox* box2;
	box2 = manage (new VBox);
	box2->pack_start (tearoff_event_box, false, false, 5);

	pack_start (contents);
	pack_start (*box2, false, false, 2);

}

TearOff::~TearOff ()
{
	delete own_window;
}

gint
TearOff::tearoff_click (GdkEventButton* ev)
{
	remove (contents);
	window_box.pack_start (contents);
	own_window->set_name (get_name());
	close_event_box.set_name (get_name());
	own_window->show_all ();
	own_window->realize ();
	hide ();
	Detach ();
	return TRUE;
}

gint
TearOff::close_click (GdkEventButton* ev)
{
	window_box.remove (contents);
	pack_start (contents);
	reorder_child (contents, 0);
	own_window->hide ();
	show_all ();
	Attach ();
	return TRUE;
}		

gint
TearOff::window_button_press (GdkEventButton* ev)
{
	dragging = true;
	drag_x = ev->x_root;
	drag_y = ev->y_root;

	Gtk::Main::grab_add (*own_window);
#if 0
	gdk_pointer_grab (own_window->get_window(), FALSE,
			  GdkEventMask (GDK_BUTTON_PRESS_MASK|GDK_BUTTON_RELEASE_MASK|GDK_POINTER_MOTION_MASK),
			  NULL,
			  NULL,
			  ev->time);
#endif

	return TRUE;
}

gint
TearOff::window_button_release (GdkEventButton* ev)
{
	dragging = false;
//	gdk_pointer_ungrab (ev->time);
	Gtk::Main::grab_remove (*own_window);
	return TRUE;
}

gint
TearOff::window_delete_event (GdkEventAny* ev)
{
	return close_click(0);
}

gint
TearOff::window_motion (GdkEventMotion* ev)
{
	gint x;
	gint y;
	gint mx, my;
	double x_delta;
	double y_delta;
	Gdk_Window win (own_window->get_window());
	
	own_window->get_pointer (mx, my);

	if (!dragging) {
		return TRUE;
	}

	x_delta = ev->x_root - drag_x;
	y_delta = ev->y_root - drag_y;

	win.get_root_origin (x, y);
	win.move ((gint) floor (x + x_delta), (gint) floor (y + y_delta));
	
	drag_x = ev->x_root;
	drag_y = ev->y_root;
	
	return TRUE;
}

bool
TearOff::torn_off() const
{
	return own_window->is_visible();
}



[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]