LUMIERA.clone/src/gui/widgets/timeline/timeline-body.cpp

256 lines
6 KiB
C++
Raw Normal View History

2008-05-31 14:22:15 +02:00
/*
timeline.cpp - Implementation of the timeline widget
Copyright (C) Lumiera.org
2008, Joel Holdsworth <joel@airwebreathe.org.uk>
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 <cairomm-1.0/cairomm/cairomm.h>
#include <boost/foreach.hpp>
2008-05-31 14:22:15 +02:00
#include "timeline-body.hpp"
2008-05-31 18:44:44 +02:00
#include "../timeline-widget.hpp"
#include "../../window-manager.hpp"
2008-05-31 14:22:15 +02:00
using namespace Gtk;
using namespace std;
using namespace lumiera::gui;
using namespace lumiera::gui::widgets;
2008-05-31 14:22:15 +02:00
using namespace lumiera::gui::widgets::timeline;
namespace lumiera {
namespace gui {
namespace widgets {
namespace timeline {
2008-05-31 18:44:44 +02:00
TimelineBody::TimelineBody(lumiera::gui::widgets::TimelineWidget *timeline_widget) :
Glib::ObjectBase("TimelineBody"),
dragType(None),
mouseDownX(0),
mouseDownY(0),
beginShiftTimeOffset(0),
2008-05-31 14:22:15 +02:00
timelineWidget(timeline_widget)
{
REQUIRE(timelineWidget != NULL);
// Connect up some events
timelineWidget->horizontalAdjustment.signal_value_changed().connect(
sigc::mem_fun(this, &TimelineBody::on_scroll) );
timelineWidget->verticalAdjustment.signal_value_changed().connect(
sigc::mem_fun(this, &TimelineBody::on_scroll) );
// Install style properties
gtk_widget_class_install_style_property(
GTK_WIDGET_CLASS(G_OBJECT_GET_CLASS(gobj())),
g_param_spec_boxed("background",
"Track Background",
"The background colour of timeline tracks",
GDK_TYPE_COLOR, G_PARAM_READABLE));
}
2008-05-31 14:22:15 +02:00
2008-06-25 21:23:53 +02:00
void
TimelineBody::on_realize()
{
Widget::on_realize();
// We wish to receive event notifications
add_events(
Gdk::POINTER_MOTION_MASK |
Gdk::SCROLL_MASK |
Gdk::BUTTON_PRESS_MASK |
Gdk::BUTTON_RELEASE_MASK);
// Apply the cursor if possible
timelineWidget->tool->apply_cursor();
2008-06-25 21:23:53 +02:00
}
2008-05-31 14:22:15 +02:00
void
TimelineBody::on_scroll()
{
queue_draw();
}
bool
TimelineBody::on_scroll_event (GdkEventScroll* event)
{
2008-06-25 21:23:53 +02:00
REQUIRE(event != NULL);
REQUIRE(timelineWidget != NULL);
if(event->state & GDK_CONTROL_MASK)
2008-05-31 14:22:15 +02:00
{
switch(event->direction)
{
case GDK_SCROLL_UP:
// User scrolled up. Zoom in
timelineWidget->zoom_view(event->x, 1);
break;
case GDK_SCROLL_DOWN:
// User scrolled down. Zoom out
timelineWidget->zoom_view(event->x, -1);
break;
}
}
else
{
switch(event->direction)
{
case GDK_SCROLL_UP:
// User scrolled up. Shift 1/16th left
timelineWidget->shift_view(-1);
break;
case GDK_SCROLL_DOWN:
// User scrolled down. Shift 1/16th right
timelineWidget->shift_view(1);
break;
}
2008-05-31 14:22:15 +02:00
}
}
2008-06-25 21:23:53 +02:00
bool
TimelineBody::on_button_press_event(GdkEventButton* event)
{
mouseDownX = event->x;
mouseDownY = event->y;
switch(event->button)
{
case 2:
begin_shift_drag();
break;
default:
dragType = None;
break;
}
}
bool
TimelineBody::on_button_release_event(GdkEventButton* event)
{
// Terminate any drags
dragType = None;
}
2008-06-25 21:23:53 +02:00
bool
TimelineBody::on_motion_notify_event(GdkEventMotion *event)
{
2008-06-25 21:23:53 +02:00
REQUIRE(event != NULL);
switch(dragType)
{
case Shift:
{
const int64_t scale = timelineWidget->get_time_scale();
gavl_time_t offset = beginShiftTimeOffset +
(int64_t)(mouseDownX - event->x) * scale;
timelineWidget->set_time_offset(offset);
set_vertical_offset((int)(mouseDownY - event->y) +
beginShiftVerticalOffset);
break;
}
}
2008-06-25 21:23:53 +02:00
// false so that the message is passed up to the owner TimelineWidget
return false;
}
2008-05-31 14:22:15 +02:00
bool
TimelineBody::on_expose_event(GdkEventExpose* event)
{
2008-06-25 21:23:53 +02:00
REQUIRE(event != NULL);
REQUIRE(timelineWidget != NULL);
// This is where we draw on the window
Glib::RefPtr<Gdk::Window> window = get_window();
if(!window)
return false;
// Makes sure the widget styles have been loaded
read_styles();
// Prepare to render via cairo
2008-06-25 21:23:53 +02:00
Glib::RefPtr<Style> style = get_style();
const Allocation allocation = get_allocation();
Cairo::RefPtr<Cairo::Context> cairo = window->create_cairo_context();
2008-06-25 21:23:53 +02:00
REQUIRE(style);
REQUIRE(cairo);
// Translate the view by the scroll distance
cairo->translate(0, -get_vertical_offset());
// Interate drawing each track
BOOST_FOREACH( Track* track, timelineWidget->tracks )
{
ASSERT(track != NULL);
2008-05-31 14:22:15 +02:00
const int height = track->get_height();
ASSERT(height >= 0);
// Draw the track background
cairo->rectangle(0, 0, allocation.get_width(), height);
gdk_cairo_set_source_color(cairo->cobj(), &background);
cairo->fill();
// Render the track
cairo->save();
track->draw_track(cairo);
cairo->restore();
// Shift for the next track
cairo->translate(0, height + TimelineWidget::TrackPadding);
}
2008-05-31 18:44:44 +02:00
return true;
}
void
TimelineBody::begin_shift_drag()
{
dragType = Shift;
beginShiftTimeOffset = timelineWidget->get_time_offset();
beginShiftVerticalOffset = get_vertical_offset();
}
int
TimelineBody::get_vertical_offset() const
{
return (int)timelineWidget->verticalAdjustment.get_value();
}
void
TimelineBody::set_vertical_offset(int offset)
{
timelineWidget->verticalAdjustment.set_value(offset);
}
2008-05-31 18:44:44 +02:00
void
TimelineBody::read_styles()
{
background = WindowManager::read_style_colour_property(
*this, "background", 0, 0, 0);
}
2008-05-31 14:22:15 +02:00
} // namespace timeline
} // namespace widgets
} // namespace gui
} // namespace lumiera