/* timeline-header-container.cpp - Implementation of the timeline header container widget Copyright (C) Lumiera.org 2008, Joel Holdsworth 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. * *****************************************************/ #include #include "timeline-header-container.hpp" #include "timeline-track.hpp" #include "../timeline-widget.hpp" using namespace Gtk; using namespace std; using namespace boost; using namespace util; namespace gui { namespace widgets { namespace timeline { TimelineHeaderContainer::TimelineHeaderContainer( gui::widgets::TimelineWidget &timeline_widget) : Glib::ObjectBase("TimelineHeaderContainer"), timelineWidget(timeline_widget), margin(-1), expand_button_size(12) { // This widget will not have a window at first set_flags(Gtk::NO_WINDOW); set_redraw_on_allocate(false); // Connect to the timeline widget's vertical scroll event, // so that we get notified when the view shifts timelineWidget.verticalAdjustment.signal_value_changed().connect( sigc::mem_fun(this, &TimelineHeaderContainer::on_scroll) ); // Connect to the timeline widget's hover event, // so that we get notified when tracks are hovered on timelineWidget.hovering_track_changed_signal().connect( sigc::mem_fun(this, &TimelineHeaderContainer::on_hovering_track_changed) ); // Create the context menu Menu::MenuList& menu_list = contextMenu.items(); menu_list.push_back( Menu_Helpers::MenuElem(_("_Add Track"), sigc::mem_fun(timelineWidget, &TimelineWidget::on_add_track_command) ) ); // Install style properties register_styles(); // Load the styles up read_styles(); } void TimelineHeaderContainer::update_headers() { // Ensure headers are parented correctly pair, shared_ptr > pair; BOOST_FOREACH( pair, timelineWidget.trackMap ) { // Set the header's parent widget Widget &widget = lookup_timeline_track(pair.first)-> get_header_widget(); const Container *parent = widget.get_parent(); if(parent == NULL) // Is the header unparented? widget.set_parent(*this); ENSURE(widget.get_parent() == this); } } void TimelineHeaderContainer::on_realize() { set_flags(Gtk::NO_WINDOW); // Call base class: Gtk::Container::on_realize(); // Create the GdkWindow: GdkWindowAttr attributes; memset(&attributes, 0, sizeof(attributes)); Allocation allocation = get_allocation(); // Set initial position and size of the Gdk::Window: attributes.x = allocation.get_x(); attributes.y = allocation.get_y(); attributes.width = allocation.get_width(); attributes.height = allocation.get_height(); attributes.event_mask = get_events () | Gdk::EXPOSURE_MASK; attributes.window_type = GDK_WINDOW_CHILD; attributes.wclass = GDK_INPUT_OUTPUT; gdkWindow = Gdk::Window::create(get_window(), &attributes, GDK_WA_X | GDK_WA_Y); unset_flags(Gtk::NO_WINDOW); set_window(gdkWindow); // Unset the background so as to make the colour match the parent // window unset_bg(STATE_NORMAL); // Make the widget receive expose events gdkWindow->set_user_data(gobj()); // Make the widget sensitive to mouse events add_events( Gdk::POINTER_MOTION_MASK | Gdk::BUTTON_PRESS_MASK | Gdk::BUTTON_RELEASE_MASK); } void TimelineHeaderContainer::on_unrealize() { // Unreference any window we may have created gdkWindow.clear(); // Call base class: Container::on_unrealize(); } bool TimelineHeaderContainer::on_button_press_event ( GdkEventButton* event) { REQUIRE(event != NULL); switch(event->button) { case 1: // Left Click // Did the user press the button on an expander? if(hoveringExpander != NULL) { // Yes? The prime for a release event clickedExpander = hoveringExpander; queue_draw(); } break; case 3: // Right Click { // Popup the context menu shared_ptr header( timelineWidget.layoutHelper.header_from_point( Gdk::Point(event->x, event->y))); // Are we hovering on a header? if(header) { // Yes - show the header's context menu header->show_header_context_menu( event->button, event->time); } else { // No - show the default context menu contextMenu.popup(event->button, event->time); } break; } } return true; } bool TimelineHeaderContainer::on_button_release_event ( GdkEventButton* event) { // Did the user release the button on an expander? if(clickedExpander != NULL) { // Yes? The toggle the expanding clickedExpander->expand_collapse( clickedExpander->get_expanded() ? Track::Collapse : Track::Expand); clickedExpander.reset(); timelineWidget.layoutHelper.update_layout(); } return Container::on_button_release_event(event); } bool TimelineHeaderContainer::on_motion_notify_event ( GdkEventMotion* event) { REQUIRE(event != NULL); // Are we hovering on an expander? shared_ptr expander = expander_button_from_point( Gdk::Point(event->x, event->y)); if(expander != hoveringExpander) { hoveringExpander = expander; queue_draw(); } return Container::on_motion_notify_event(event); } void TimelineHeaderContainer::on_size_request (Requisition* requisition) { // We don't care about the size of all the child widgets, but if we // don't send the size request down the tree, some widgets fail to // calculate their text layout correctly. const TimelineLayoutHelper::TrackTree &layout_tree = timelineWidget.layoutHelper.get_layout_tree(); TimelineLayoutHelper::TrackTree::pre_order_iterator iterator; for(iterator = ++layout_tree.begin(); // ++ so that we skip the sequence root iterator != layout_tree.end(); iterator++) { Widget &widget = lookup_timeline_track(*iterator)->get_header_widget(); if(widget.is_visible()) widget.size_request(); } // Initialize the output parameter: *requisition = Gtk::Requisition(); requisition->width = TimelineWidget::HeaderWidth; requisition->height = 0; } void TimelineHeaderContainer::on_size_allocate (Allocation& allocation) { // Use the offered allocation for this container: set_allocation(allocation); // Resize the widget's window if(gdkWindow) { gdkWindow->move(allocation.get_x(), allocation.get_y()); gdkWindow->resize( allocation.get_width(), allocation.get_height()); } // Relayout the child widgets of the headers layout_headers(); } void TimelineHeaderContainer::forall_vfunc(gboolean /* include_internals */, GtkCallback callback, gpointer callback_data) { REQUIRE(callback != NULL); BOOST_FOREACH( shared_ptr track, get_tracks() ) { REQUIRE(track); forall_vfunc_recursive(track, callback, callback_data); } } void TimelineHeaderContainer::on_remove(Widget*) { // Do nothing - this is just to keep Gtk::Container happy } void TimelineHeaderContainer::on_layout_changed() { layout_headers(); } bool TimelineHeaderContainer::on_expose_event(GdkEventExpose *event) { if(gdkWindow) { const Allocation container_allocation = get_allocation(); // Paint a border underneath all the headers const TimelineLayoutHelper::TrackTree &layout_tree = timelineWidget.layoutHelper.get_layout_tree(); TimelineLayoutHelper::TrackTree::pre_order_iterator iterator; for(iterator = ++layout_tree.begin(); // ++ so that we skip the sequence root iterator != layout_tree.end(); iterator++) { shared_ptr model_track = *iterator; REQUIRE(model_track); draw_header_decoration(model_track, Gdk::Rectangle(0, 0, container_allocation.get_width(), container_allocation.get_height())); } } return Container::on_expose_event(event); } void TimelineHeaderContainer::on_scroll() { // If the scroll has changed, we will have to shift all the // header widgets layout_headers(); } void TimelineHeaderContainer::on_hovering_track_changed( boost::shared_ptr hovering_track) { (void)hovering_track; // The hovering track has changed, redraw so we can light the header queue_draw(); } void TimelineHeaderContainer::layout_headers() { REQUIRE(margin >= 0); // read_styles must have been called before now // We can't layout before the widget has been set up if(!gdkWindow) return; TimelineLayoutHelper &layout_helper = timelineWidget.layoutHelper; const TimelineLayoutHelper::TrackTree &layout_tree = layout_helper.get_layout_tree(); TimelineLayoutHelper::TrackTree::pre_order_iterator iterator; for(iterator = ++layout_tree.begin(); // ++ so that we skip the sequence root iterator != layout_tree.end(); iterator++) { const shared_ptr timeline_track = lookup_timeline_track(*iterator); Widget &widget = timeline_track->get_header_widget(); optional header_rect = layout_helper.get_track_header_rect(timeline_track); if(header_rect) { REQUIRE(header_rect->get_width() >= 0); REQUIRE(header_rect->get_height() >= 0); // Calculate the allocation of the header widget Allocation header_allocation( header_rect->get_x() + margin + expand_button_size, // x header_rect->get_y() + margin, // y max( header_rect->get_width() - expand_button_size - margin * 2, 0 ), // width header_rect->get_height() - margin * 2); // height // Apply the allocation to the header widget.size_allocate (header_allocation); if(!widget.is_visible()) widget.show(); } else // No header rect, so the track must be hidden if(widget.is_visible()) widget.hide(); } // Repaint the background of our parenting queue_draw (); } void TimelineHeaderContainer::forall_vfunc_recursive( shared_ptr model_track, GtkCallback callback, gpointer callback_data) { REQUIRE(callback != NULL); callback( lookup_timeline_track(model_track)-> get_header_widget().gobj(), callback_data) ; // Recurse through all the children BOOST_FOREACH( shared_ptr child, model_track->get_child_tracks() ) forall_vfunc_recursive(child, callback, callback_data); } void TimelineHeaderContainer::draw_header_decoration( shared_ptr model_track, const Gdk::Rectangle &clip_rect) { REQUIRE(model_track != NULL); REQUIRE(clip_rect.get_width() > 0); REQUIRE(clip_rect.get_height() > 0); Glib::RefPtr