LUMIERA.clone/src/gui/widgets/timeline/timeline-track.cpp
Ichthyostega 7be1b7d35d Switch from TR1 preveiw to the new standard headers
- functional
- memory
- unordered collections
2014-04-03 22:42:48 +02:00

339 lines
7.9 KiB
C++

/*
timeline-track.cpp - Implementation of the timeline track object
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-track.hpp"
#include "gui/widgets/timeline-widget.hpp"
#include "gui/window-manager.hpp"
#include "gui/dialogs/name-chooser.hpp"
#include "include/logging.h"
using namespace Gtk;
using namespace sigc;
using std::shared_ptr;
namespace gui {
namespace widgets {
namespace timeline {
const float Track::ExpandAnimationPeriod = 0.15;
Track::Track(TimelineWidget &timeline_widget,
shared_ptr<model::Track> track) :
timelineWidget(timeline_widget),
modelTrack(track),
expanded(true),
expandDirection(None),
headerWidget(*this),
enableButton(Gtk::StockID("track_enabled"), WindowManager::MenuIconSize),
lockButton(Gtk::StockID("track_unlocked"), WindowManager::MenuIconSize)
{
REQUIRE(modelTrack);
titleMenuButton.set_relief(RELIEF_HALF);
titleMenuButton.unset_flags(CAN_FOCUS);
buttonBar.append(enableButton, mem_fun(this, &Track::onToggleEnabled));
buttonBar.append(lockButton, mem_fun(this, &Track::onToggleLocked));
headerWidget.set_child_widget(headerBox);
headerBox.pack_start(titleMenuButton, PACK_SHRINK);
headerBox.pack_start(buttonBar, PACK_SHRINK);
headerWidget.show_all();
// Setup the title menu button
Menu::MenuList& title_list = titleMenuButton.get_menu().items();
title_list.push_back( Menu_Helpers::MenuElem(_("_Name..."),
mem_fun(this, &Track::on_set_name) ) );
title_list.push_back( Menu_Helpers::SeparatorElem() );
title_list.push_back( Menu_Helpers::MenuElem(_("_Remove"),
mem_fun(this, &Track::on_remove_track) ) );
updateEnableButton();
updateLockButton();
updateName();
// Setup the context menu
Menu::MenuList& context_list = contextMenu.items();
//context_list.push_back( Menu_Helpers::MenuElem(_("_Add Track"),
// mem_fun(timelineWidget, &TimelineWidget::on_add_track_command) ) );
context_list.push_back( Menu_Helpers::MenuElem(_("_Remove Track"),
mem_fun(this, &Track::on_remove_track) ) );
// Connect to the model
modelTrack->signalEnabledChanged().connect(sigc::mem_fun(this,
&Track::onEnabledChanged));
modelTrack->signalLockedChanged().connect(sigc::mem_fun(this,
&Track::onLockedChanged));
modelTrack->signalNameChanged().connect(sigc::mem_fun(this,
&Track::onNameChanged));
}
Track::~Track()
{
headerWidget.remove(headerBox);
}
Gtk::Widget&
Track::get_header_widget()
{
return headerWidget;
}
shared_ptr<model::Track>
Track::getModelTrack() const
{
return modelTrack;
}
int
Track::get_height() const
{
return 100;
}
bool
Track::get_expanded() const
{
return expanded;
}
shared_ptr<timeline::Clip>
Track::getClipAt(Time) const
{
// Default implementation returns empty pointer
return shared_ptr<timeline::Clip>();
}
void
Track::expand_collapse(ExpandDirection direction)
{
REQUIRE(direction != None);
expandDirection = direction;
if(direction == Expand)
{
expanded = true;
expandAnimationState = 0.0;
}
else
{
expanded = false;
expandAnimationState = 1.0;
}
// Create a timer if we don't already have one
if(!expand_timer)
{
expand_timer.reset(new Glib::Timer());
expand_timer->start();
}
else // Reset the timer if we do
expand_timer->reset();
}
float
Track::get_expand_animation_state() const
{
ENSURE(expandAnimationState >= 0.0 &&
expandAnimationState <= 1.0);
return expandAnimationState;
}
bool
Track::is_expand_animating() const
{
return expandDirection != None;
}
void
Track::tick_expand_animation()
{
REQUIRE(expandDirection != None); // tick_expand_animation should not
// be unless is_expand_animating
// returns true
REQUIRE(expand_timer);
const float delta =
(float)expand_timer->elapsed() / ExpandAnimationPeriod;
expand_timer->reset(); // reset the timer to t=0
if(expandDirection == Expand)
{
expandAnimationState += delta;
if(expandAnimationState >= 1.0)
expandDirection = None;
}
else
{
expandAnimationState -= delta;
if(expandAnimationState <= 0.0)
expandDirection = None;
}
if(expandDirection == None)
expand_timer.reset(); // We've finished with the timer - delete it
}
Gtk::ExpanderStyle
Track::get_expander_style() const
{
if(expanded)
{
if(expandDirection == None)
return EXPANDER_EXPANDED;
else if(expandAnimationState >= 2.0 / 3.0)
return EXPANDER_SEMI_EXPANDED;
else if(expandAnimationState >= 1.0 / 3.0)
return EXPANDER_SEMI_COLLAPSED;
else
return EXPANDER_COLLAPSED;
}
else
{
if(expandDirection == None)
return EXPANDER_COLLAPSED;
else if(expandAnimationState >= 2.0 / 3.0)
return EXPANDER_EXPANDED;
else if(expandAnimationState >= 1.0 / 3.0)
return EXPANDER_SEMI_EXPANDED;
else
return EXPANDER_SEMI_COLLAPSED;
}
NOBUG_ERROR(gui, "Track::get_expander_style() final return reached");
return EXPANDER_COLLAPSED; // This should never happen
}
void
Track::show_header_context_menu(guint button, guint32 time)
{
contextMenu.popup(button, time);
}
void
Track::onEnabledChanged(bool)
{
updateEnableButton();
}
void
Track::onLockedChanged(bool)
{
updateLockButton();
}
void
Track::on_set_name()
{
REQUIRE(modelTrack);
Gtk::Window *window = dynamic_cast<Window*>(
timelineWidget.get_toplevel());
REQUIRE(window != NULL);
dialogs::NameChooser dialog(*window,
_("Set Track Name"), modelTrack->get_name());
if(dialog.run() == RESPONSE_OK)
modelTrack->set_name(dialog.get_name());
}
void
Track::onNameChanged(std::string)
{
updateName();
}
void
Track::on_remove_track()
{
REQUIRE(modelTrack);
shared_ptr<TimelineState> state = timelineWidget.get_state();
REQUIRE(state);
state->get_sequence()->remove_descendant_track(modelTrack);
}
void
Track::onToggleEnabled()
{
bool status = modelTrack->getEnabled();
modelTrack->setEnabled(!status);
}
void
Track::onToggleLocked()
{
bool status = modelTrack->getLocked();
modelTrack->setLocked(!status);
}
void
Track::updateEnableButton()
{
REQUIRE (modelTrack);
if (modelTrack->getEnabled())
{
enableButton.set_stock_id(Gtk::StockID("track_enabled"), WindowManager::MenuIconSize);
enableButton.set_tooltip_text(_("Disable track"));
}
else
{
enableButton.set_stock_id(Gtk::StockID("track_disabled"), WindowManager::MenuIconSize);
enableButton.set_tooltip_text(_("Enable track"));
}
}
void
Track::updateLockButton()
{
REQUIRE (modelTrack);
if (modelTrack->getLocked())
{
lockButton.set_stock_id(Gtk::StockID("track_locked"), WindowManager::MenuIconSize);
lockButton.set_tooltip_text(_("Unlock track"));
}
else
{
lockButton.set_stock_id(Gtk::StockID("track_unlocked"), WindowManager::MenuIconSize);
lockButton.set_tooltip_text(_("Lock track"));
}
}
void
Track::updateName()
{
REQUIRE(modelTrack);
titleMenuButton.set_label(modelTrack->get_name());
}
} // namespace timeline
} // namespace widgets
} // namespace gui