lumiera_/src/gui/widgets/timeline-widget.cpp

292 lines
7.2 KiB
C++
Raw Normal View History

2008-04-19 21:31:27 +02:00
/*
2008-05-31 14:22:15 +02:00
timeline-widget.cpp - Implementation of the timeline widget
2008-04-19 21:31:27 +02:00
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 "timeline-widget.hpp"
#include "timeline/timeline-arrow-tool.hpp"
#include "timeline/timeline-ibeam-tool.hpp"
2008-04-19 21:31:27 +02:00
#include <boost/foreach.hpp>
2008-05-22 20:19:04 +02:00
using namespace Gtk;
2008-05-31 14:22:15 +02:00
using namespace std;
using namespace lumiera::gui::widgets::timeline;
2008-05-22 20:19:04 +02:00
2008-04-19 21:31:27 +02:00
namespace lumiera {
namespace gui {
namespace widgets {
const int TimelineWidget::TrackPadding = 1;
const int TimelineWidget::HeaderWidth = 100;
const double TimelineWidget::ZoomIncrement = 1.25;
const int64_t TimelineWidget::MaxScale = 30000000;
2008-05-22 20:19:04 +02:00
TimelineWidget::TimelineWidget() :
Table(2, 2),
2008-06-23 18:07:57 +02:00
timeOffset(0),
timeScale(1),
2008-05-31 14:22:15 +02:00
totalHeight(0),
horizontalAdjustment(0, 0, 0),
verticalAdjustment(0, 0, 0),
2008-05-22 20:19:04 +02:00
horizontalScroll(horizontalAdjustment),
verticalScroll(verticalAdjustment),
tool(NULL)
2008-06-19 22:57:53 +02:00
{
body = new TimelineBody(this);
ENSURE(body != NULL);
2008-06-19 22:57:53 +02:00
headerContainer = new HeaderContainer(this);
ENSURE(headerContainer != NULL);
2008-06-19 22:57:53 +02:00
horizontalAdjustment.signal_value_changed().connect( sigc::mem_fun(
this, &TimelineWidget::on_scroll) );
verticalAdjustment.signal_value_changed().connect( sigc::mem_fun(
this, &TimelineWidget::on_scroll) );
body->signal_motion_notify_event().connect( sigc::mem_fun(
this, &TimelineWidget::on_motion_in_body_notify_event) );
2008-07-24 00:23:48 +02:00
set_time_scale(GAVL_TIME_SCALE / 200);
2008-04-19 21:31:27 +02:00
2008-06-19 22:57:53 +02:00
attach(*body, 1, 2, 1, 2, FILL|EXPAND, FILL|EXPAND);
attach(ruler, 1, 2, 0, 1, FILL|EXPAND, SHRINK);
attach(*headerContainer, 0, 1, 1, 2, SHRINK, FILL|EXPAND);
attach(horizontalScroll, 1, 2, 2, 3, FILL|EXPAND, SHRINK);
attach(verticalScroll, 2, 3, 1, 2, SHRINK, FILL|EXPAND);
2008-05-31 14:22:15 +02:00
2008-06-19 22:57:53 +02:00
tracks.push_back(&video1);
tracks.push_back(&video2);
update_tracks();
set_tool(timeline::Arrow);
2008-06-19 22:57:53 +02:00
}
2008-05-31 14:22:15 +02:00
TimelineWidget::~TimelineWidget()
2008-06-19 22:57:53 +02:00
{
REQUIRE(body != NULL);
if(body != NULL)
body->unreference();
REQUIRE(headerContainer != NULL);
if(headerContainer != NULL)
headerContainer->unreference();
REQUIRE(tool != NULL);
if(tool != NULL)
delete tool;
2008-06-19 22:57:53 +02:00
}
2008-05-31 14:22:15 +02:00
gavl_time_t
TimelineWidget::get_time_offset() const
{
2008-06-23 18:07:57 +02:00
return timeOffset;
}
void
TimelineWidget::set_time_offset(gavl_time_t time_offset)
{
2008-06-23 18:07:57 +02:00
timeOffset = time_offset;
horizontalAdjustment.set_value(time_offset);
ruler.set_time_offset(time_offset);
}
int64_t
TimelineWidget::get_time_scale() const
{
return timeScale;
}
void
TimelineWidget::set_time_scale(int64_t time_scale)
{
timeScale = time_scale;
ruler.set_time_scale(time_scale);
const int view_width = body->get_allocation().get_width();
horizontalAdjustment.set_page_size(timeScale * view_width);
}
void
TimelineWidget::zoom_view(int zoom_size)
{
const Allocation allocation = body->get_allocation();
zoom_view(allocation.get_width() / 2, zoom_size);
}
void
TimelineWidget::zoom_view(int point, int zoom_size)
{
int64_t new_time_scale = (double)timeScale * pow(1.25, -zoom_size);
// Limit zooming in too close
if(new_time_scale < 1) new_time_scale = 1;
// Nudge zoom problems caused by integer rounding
if(new_time_scale == timeScale && zoom_size < 0)
new_time_scale++;
// Limit zooming out too far
if(new_time_scale > MaxScale)
new_time_scale = MaxScale;
// The view must be shifted so that the zoom is centred on the cursor
set_time_offset(get_time_offset() +
(timeScale - new_time_scale) * point);
// Apply the new scale
set_time_scale(new_time_scale);
}
void
TimelineWidget::shift_view(int shift_size)
{
const int view_width = body->get_allocation().get_width();
set_time_offset(get_time_offset() +
shift_size * timeScale * view_width / 16);
}
ToolType
TimelineWidget::get_tool() const
{
REQUIRE(tool != NULL);
if(tool != NULL)
return tool->get_type();
return None;
}
void
TimelineWidget::set_tool(ToolType tool_type)
{
// Tidy up old tool
if(tool != NULL)
{
// Do we need to change tools?
2008-08-04 17:40:38 +02:00
if(tool->get_type() == tool_type)
return;
delete tool;
}
// Create the new tool
switch(tool_type)
{
case timeline::Arrow:
tool = new timeline::ArrowTool(this);
break;
case timeline::IBeam:
tool = new timeline::IBeamTool(this);
break;
}
// Apply the cursor if possible
tool->apply_cursor();
}
2008-05-31 14:22:15 +02:00
void
TimelineWidget::on_scroll()
2008-06-19 22:57:53 +02:00
{
2008-06-23 18:07:57 +02:00
timeOffset = horizontalAdjustment.get_value();
ruler.set_time_offset(timeOffset);
2008-06-19 22:57:53 +02:00
}
void
TimelineWidget::on_size_allocate(Allocation& allocation)
2008-06-19 22:57:53 +02:00
{
Widget::on_size_allocate(allocation);
2008-06-19 22:57:53 +02:00
update_scroll();
}
2008-05-31 14:22:15 +02:00
void
TimelineWidget::update_tracks()
2008-06-19 22:57:53 +02:00
{
ASSERT(headerContainer != NULL);
headerContainer->update_headers();
// Recalculate the total height of the timeline scrolled area
totalHeight = 0;
BOOST_FOREACH( Track* track, tracks )
{
ASSERT(track != NULL);
totalHeight += track->get_height() + TrackPadding;
}
}
2008-05-31 14:22:15 +02:00
void
TimelineWidget::update_scroll()
2008-06-19 22:57:53 +02:00
{
ASSERT(body != NULL);
const Allocation body_allocation = body->get_allocation();
//----- Horizontal Scroll ------//
// TEST CODE
horizontalAdjustment.set_upper(1000 * GAVL_TIME_SCALE / 200);
horizontalAdjustment.set_lower(-1000 * GAVL_TIME_SCALE / 200);
// Set the page size
horizontalAdjustment.set_page_size(
timeScale * body_allocation.get_width());
2008-06-19 22:57:53 +02:00
//----- Vertical Scroll -----//
// Calculate the vertical length that can be scrolled:
// the total height of all the tracks minus one screenful
int y_scroll_length = totalHeight - body_allocation.get_height();
if(y_scroll_length < 0) y_scroll_length = 0;
// If by resizing we're now over-scrolled, scroll back to
// maximum distance
if((int)verticalAdjustment.get_value() > y_scroll_length)
verticalAdjustment.set_value(y_scroll_length);
verticalAdjustment.set_upper(y_scroll_length);
// Hide the scrollbar if no scrolling is possible
2008-07-24 00:23:48 +02:00
#if 0
// Having this code included seems to cause a layout loop as the
// window is shrunk
if(y_scroll_length <= 0 && verticalScroll.is_visible())
2008-06-19 22:57:53 +02:00
verticalScroll.hide();
2008-07-24 00:23:48 +02:00
else if(y_scroll_length > 0 && !verticalScroll.is_visible())
2008-06-19 22:57:53 +02:00
verticalScroll.show();
2008-07-24 00:23:48 +02:00
#endif
2008-06-19 22:57:53 +02:00
}
2008-05-31 14:22:15 +02:00
int
TimelineWidget::get_y_scroll_offset() const
2008-06-19 22:57:53 +02:00
{
return (int)verticalAdjustment.get_value();
}
2008-04-19 21:31:27 +02:00
bool
TimelineWidget::on_motion_in_body_notify_event(GdkEventMotion *event)
2008-06-25 21:23:53 +02:00
{
REQUIRE(event != NULL);
ruler.set_mouse_chevron_offset(event->x);
return true;
2008-06-25 21:23:53 +02:00
}
2008-04-19 21:31:27 +02:00
} // namespace widgets
} // namespace gui
} // namespace lumiera