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>
|
2008-06-07 14:53:17 +02:00
|
|
|
#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"
|
2008-06-19 00:57:47 +02:00
|
|
|
#include "../../window-manager.hpp"
|
2008-05-31 14:22:15 +02:00
|
|
|
|
2008-08-07 21:27:41 +02:00
|
|
|
#include "timeline-arrow-tool.hpp"
|
|
|
|
|
#include "timeline-ibeam-tool.hpp"
|
|
|
|
|
|
2008-05-31 14:22:15 +02:00
|
|
|
using namespace Gtk;
|
|
|
|
|
using namespace std;
|
2008-06-19 00:57:47 +02:00
|
|
|
using namespace lumiera::gui;
|
2008-05-31 19:21:05 +02:00
|
|
|
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-08-07 21:27:41 +02:00
|
|
|
TimelineBody::TimelineBody(lumiera::gui::widgets::TimelineWidget
|
|
|
|
|
*timeline_widget) :
|
2008-05-31 18:44:44 +02:00
|
|
|
Glib::ObjectBase("TimelineBody"),
|
2008-08-07 21:27:41 +02:00
|
|
|
tool(NULL),
|
2008-07-16 23:33:42 +02:00
|
|
|
dragType(None),
|
|
|
|
|
mouseDownX(0),
|
|
|
|
|
mouseDownY(0),
|
|
|
|
|
beginShiftTimeOffset(0),
|
2008-08-14 00:47:47 +02:00
|
|
|
selectionAlpha(0.5),
|
2008-05-31 14:22:15 +02:00
|
|
|
timelineWidget(timeline_widget)
|
2008-06-23 11:54:37 +02:00
|
|
|
{
|
|
|
|
|
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
|
2008-08-14 00:47:47 +02:00
|
|
|
register_styles();
|
2008-06-23 11:54:37 +02:00
|
|
|
}
|
2008-05-31 14:22:15 +02:00
|
|
|
|
2008-08-07 21:27:41 +02:00
|
|
|
TimelineBody::~TimelineBody()
|
|
|
|
|
{
|
|
|
|
|
REQUIRE(tool != NULL);
|
|
|
|
|
if(tool != NULL)
|
|
|
|
|
delete tool;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ToolType
|
|
|
|
|
TimelineBody::get_tool() const
|
|
|
|
|
{
|
|
|
|
|
REQUIRE(tool != NULL);
|
|
|
|
|
if(tool != NULL)
|
|
|
|
|
return tool->get_type();
|
|
|
|
|
return lumiera::gui::widgets::timeline::None;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
TimelineBody::set_tool(timeline::ToolType tool_type)
|
|
|
|
|
{
|
|
|
|
|
// Tidy up old tool
|
|
|
|
|
if(tool != NULL)
|
|
|
|
|
{
|
|
|
|
|
// Do we need to change tools?
|
|
|
|
|
if(tool->get_type() == tool_type)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
delete tool;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Create the new tool
|
|
|
|
|
switch(tool_type)
|
|
|
|
|
{
|
|
|
|
|
case timeline::Arrow:
|
|
|
|
|
tool = new ArrowTool(this);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case timeline::IBeam:
|
|
|
|
|
tool = new IBeamTool(this);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Apply the cursor if possible
|
|
|
|
|
tool->apply_cursor();
|
|
|
|
|
}
|
|
|
|
|
|
2008-06-25 21:23:53 +02:00
|
|
|
void
|
|
|
|
|
TimelineBody::on_realize()
|
|
|
|
|
{
|
|
|
|
|
Widget::on_realize();
|
|
|
|
|
|
2008-08-04 17:39:36 +02:00
|
|
|
// We wish to receive event notifications
|
2008-07-16 23:33:42 +02:00
|
|
|
add_events(
|
|
|
|
|
Gdk::POINTER_MOTION_MASK |
|
|
|
|
|
Gdk::SCROLL_MASK |
|
|
|
|
|
Gdk::BUTTON_PRESS_MASK |
|
|
|
|
|
Gdk::BUTTON_RELEASE_MASK);
|
2008-08-04 17:39:36 +02:00
|
|
|
|
|
|
|
|
// Apply the cursor if possible
|
2008-08-07 21:27:41 +02:00
|
|
|
tool->apply_cursor();
|
2008-06-25 21:23:53 +02:00
|
|
|
}
|
|
|
|
|
|
2008-05-31 14:22:15 +02:00
|
|
|
void
|
|
|
|
|
TimelineBody::on_scroll()
|
2008-06-23 11:54:37 +02:00
|
|
|
{
|
|
|
|
|
queue_draw();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
TimelineBody::on_scroll_event (GdkEventScroll* event)
|
|
|
|
|
{
|
2008-06-25 21:23:53 +02:00
|
|
|
REQUIRE(event != NULL);
|
2008-06-23 11:54:37 +02:00
|
|
|
REQUIRE(timelineWidget != NULL);
|
|
|
|
|
|
2008-06-23 12:17:19 +02:00
|
|
|
if(event->state & GDK_CONTROL_MASK)
|
2008-05-31 14:22:15 +02:00
|
|
|
{
|
2008-06-23 12:17:19 +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-23 11:54:37 +02:00
|
|
|
}
|
2008-06-25 21:23:53 +02:00
|
|
|
|
2008-07-16 23:33:42 +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;
|
2008-08-07 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Forward the event to the tool
|
|
|
|
|
tool->on_button_press_event(event);
|
|
|
|
|
|
|
|
|
|
return true;
|
2008-07-16 23:33:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
TimelineBody::on_button_release_event(GdkEventButton* event)
|
|
|
|
|
{
|
|
|
|
|
// Terminate any drags
|
|
|
|
|
dragType = None;
|
2008-08-07 21:27:41 +02:00
|
|
|
|
|
|
|
|
// Forward the event to the tool
|
|
|
|
|
tool->on_button_release_event(event);
|
|
|
|
|
|
|
|
|
|
return true;
|
2008-07-16 23:33:42 +02:00
|
|
|
}
|
|
|
|
|
|
2008-06-25 21:23:53 +02:00
|
|
|
bool
|
|
|
|
|
TimelineBody::on_motion_notify_event(GdkEventMotion *event)
|
2008-08-07 21:27:41 +02:00
|
|
|
{
|
2008-06-25 21:23:53 +02:00
|
|
|
REQUIRE(event != NULL);
|
2008-07-17 20:07:38 +02:00
|
|
|
|
2008-07-16 23:33:42 +02:00
|
|
|
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
|
|
|
|
2008-08-07 21:27:41 +02:00
|
|
|
// Forward the event to the tool
|
|
|
|
|
tool->on_motion_notify_event(event);
|
|
|
|
|
|
2008-07-17 20:07:38 +02:00
|
|
|
// false so that the message is passed up to the owner TimelineWidget
|
|
|
|
|
return false;
|
2008-06-23 11:54:37 +02:00
|
|
|
}
|
2008-05-31 14:22:15 +02:00
|
|
|
|
|
|
|
|
bool
|
|
|
|
|
TimelineBody::on_expose_event(GdkEventExpose* event)
|
2008-06-23 11:54:37 +02:00
|
|
|
{
|
2008-08-07 21:27:41 +02:00
|
|
|
Cairo::Matrix view_matrix;
|
|
|
|
|
|
2008-06-25 21:23:53 +02:00
|
|
|
REQUIRE(event != NULL);
|
|
|
|
|
REQUIRE(timelineWidget != NULL);
|
|
|
|
|
|
2008-06-23 11:54:37 +02:00
|
|
|
// 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();
|
2008-07-14 21:16:11 +02:00
|
|
|
const Allocation allocation = get_allocation();
|
2008-08-14 00:47:47 +02:00
|
|
|
Cairo::RefPtr<Cairo::Context> cr = window->create_cairo_context();
|
2008-06-25 21:23:53 +02:00
|
|
|
|
|
|
|
|
REQUIRE(style);
|
2008-08-14 00:47:47 +02:00
|
|
|
REQUIRE(cr);
|
2008-06-25 21:23:53 +02:00
|
|
|
|
2008-06-23 11:54:37 +02:00
|
|
|
// Translate the view by the scroll distance
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->translate(0, -get_vertical_offset());
|
|
|
|
|
cr->get_matrix(view_matrix);
|
2008-06-23 11:54:37 +02:00
|
|
|
|
|
|
|
|
// Interate drawing each track
|
|
|
|
|
BOOST_FOREACH( Track* track, timelineWidget->tracks )
|
|
|
|
|
{
|
|
|
|
|
ASSERT(track != NULL);
|
2008-05-31 14:22:15 +02:00
|
|
|
|
2008-06-23 11:54:37 +02:00
|
|
|
const int height = track->get_height();
|
|
|
|
|
ASSERT(height >= 0);
|
|
|
|
|
|
|
|
|
|
// Draw the track background
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->rectangle(0, 0, allocation.get_width(), height);
|
|
|
|
|
gdk_cairo_set_source_color(cr->cobj(), &backgroundColour);
|
|
|
|
|
cr->fill();
|
2008-06-23 11:54:37 +02:00
|
|
|
|
|
|
|
|
// Render the track
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->save();
|
|
|
|
|
track->draw_track(cr);
|
|
|
|
|
cr->restore();
|
2008-06-07 14:53:17 +02:00
|
|
|
|
2008-06-23 11:54:37 +02:00
|
|
|
// Shift for the next track
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->translate(0, height + TimelineWidget::TrackPadding);
|
2008-08-07 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//----- Draw the selection -----//
|
|
|
|
|
const int start_x = timelineWidget->time_to_x(
|
|
|
|
|
timelineWidget->get_selection_start());
|
|
|
|
|
const int end_x = timelineWidget->time_to_x(
|
|
|
|
|
timelineWidget->get_selection_end());
|
2008-05-31 18:44:44 +02:00
|
|
|
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->set_matrix(view_matrix);
|
2008-08-07 21:27:41 +02:00
|
|
|
|
|
|
|
|
// Draw the cover
|
|
|
|
|
if(end_x > 0 && start_x < allocation.get_width())
|
|
|
|
|
{
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->set_source_rgba(
|
|
|
|
|
(float)selectionColour.red / 0xFFFF,
|
|
|
|
|
(float)selectionColour.green / 0xFFFF,
|
|
|
|
|
(float)selectionColour.blue / 0xFFFF,
|
|
|
|
|
selectionAlpha);
|
|
|
|
|
cr->rectangle(start_x + 0.5, 0,
|
2008-08-07 21:27:41 +02:00
|
|
|
end_x - start_x, allocation.get_height());
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->fill();
|
2008-08-07 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
|
2008-08-14 00:47:47 +02:00
|
|
|
gdk_cairo_set_source_color(cr->cobj(), &selectionColour);
|
|
|
|
|
cr->set_line_width(1);
|
2008-08-07 21:27:41 +02:00
|
|
|
|
|
|
|
|
// Draw the start
|
|
|
|
|
if(start_x >= 0 && start_x < allocation.get_width())
|
|
|
|
|
{
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->move_to(start_x + 0.5, 0);
|
|
|
|
|
cr->line_to(start_x + 0.5, allocation.get_height());
|
|
|
|
|
cr->stroke_preserve();
|
2008-08-07 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Draw the end
|
|
|
|
|
if(end_x >= 0 && end_x < allocation.get_width())
|
|
|
|
|
{
|
2008-08-14 00:47:47 +02:00
|
|
|
cr->move_to(end_x + 0.5, 0);
|
|
|
|
|
cr->line_to(end_x + 0.5, allocation.get_height());
|
|
|
|
|
cr->stroke_preserve();
|
2008-08-07 21:27:41 +02:00
|
|
|
}
|
|
|
|
|
|
2008-06-23 11:54:37 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
2008-07-16 23:33:42 +02:00
|
|
|
|
|
|
|
|
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-08-14 00:47:47 +02:00
|
|
|
|
|
|
|
|
void
|
|
|
|
|
TimelineBody::register_styles() const
|
|
|
|
|
{
|
|
|
|
|
GtkWidgetClass *klass = GTK_WIDGET_CLASS(G_OBJECT_GET_CLASS(gobj()));
|
|
|
|
|
|
|
|
|
|
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));
|
|
|
|
|
|
|
|
|
|
gtk_widget_class_install_style_property(
|
|
|
|
|
GTK_WIDGET_CLASS(G_OBJECT_GET_CLASS(gobj())),
|
|
|
|
|
g_param_spec_boxed("selection",
|
|
|
|
|
"End lines of a selection",
|
|
|
|
|
"The colour of selection limit lines",
|
|
|
|
|
GDK_TYPE_COLOR, G_PARAM_READABLE));
|
|
|
|
|
|
|
|
|
|
gtk_widget_class_install_style_property(klass,
|
|
|
|
|
g_param_spec_float("selection_alpha",
|
|
|
|
|
"Selection Alpha",
|
|
|
|
|
"The transparency of the selection marque.",
|
|
|
|
|
0, 1.0, 0.5, G_PARAM_READABLE));
|
|
|
|
|
}
|
2008-05-31 18:44:44 +02:00
|
|
|
|
|
|
|
|
void
|
|
|
|
|
TimelineBody::read_styles()
|
2008-06-23 11:54:37 +02:00
|
|
|
{
|
2008-08-14 00:47:47 +02:00
|
|
|
backgroundColour = WindowManager::read_style_colour_property(
|
2008-06-23 11:54:37 +02:00
|
|
|
*this, "background", 0, 0, 0);
|
2008-08-14 00:47:47 +02:00
|
|
|
selectionColour = WindowManager::read_style_colour_property(
|
|
|
|
|
*this, "selection", 0, 0, 0);
|
|
|
|
|
|
|
|
|
|
get_style_property("selection_alpha", selectionAlpha);
|
2008-06-23 11:54:37 +02:00
|
|
|
}
|
2008-05-31 14:22:15 +02:00
|
|
|
|
|
|
|
|
} // namespace timeline
|
|
|
|
|
} // namespace widgets
|
|
|
|
|
} // namespace gui
|
|
|
|
|
} // namespace lumiera
|