+/* === S Y N F I G ========================================================= */
+/*! \file canvasview.cpp
+** \brief Template File
+**
+** $Id$
+**
+** \legal
+** Copyright (c) 2002-2005 Robert B. Quattlebaum Jr., Adrian Bentley
+** Copyright (c) 2007, 2008 Chris Moore
+** Copyright (c) 2009 Carlos López
+**
+** This package 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 package 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.
+** \endlegal
+*/
+/* ========================================================================= */
+
+/* === H E A D E R S ======================================================= */
+
+#ifdef USING_PCH
+# include "pch.h"
+#else
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
+
+#include <sigc++/adaptors/hide.h>
+
+#include <ETL/clock>
+#include <sstream>
+
+#include <gtkmm/paned.h>
+#include <gtkmm/scale.h>
+#include <gtkmm/dialog.h>
+#include <gtkmm/messagedialog.h>
+#include <gtkmm/treemodelsort.h>
+#include <gtkmm/buttonbox.h>
+#include <gtkmm/separator.h>
+
+#include <gtk/gtktreestore.h>
+#include <gtk/gtkversion.h>
+
+#include <synfig/valuenode_reference.h>
+#include <synfig/valuenode_subtract.h>
+#include <synfig/valuenode_linear.h>
+#include <synfig/valuenode_timedswap.h>
+#include <synfig/valuenode_scale.h>
+#include <synfig/valuenode_dynamiclist.h>
+#include <synfig/valuenode_twotone.h>
+#include <synfig/valuenode_stripes.h>
+#include <synfig/valuenode_blinecalctangent.h>
+#include <synfig/valuenode_blinecalcvertex.h>
+#include <synfig/valuenode_blinecalcwidth.h>
+#include <synfig/valuenode_bline.h>
+#include <synfig/layer.h>
+
+#include <synfigapp/uimanager.h>
+#include <synfigapp/canvasinterface.h>
+#include <synfigapp/selectionmanager.h>
+//#include <synfigapp/action_setwaypoint.h>
+//#include <synfigapp/action_deletewaypoint.h>
+
+#include <sigc++/retype_return.h>
+#include <sigc++/retype.h>
+//#include <sigc++/hide.h>
+
+#include "canvasview.h"
+#include "instance.h"
+#include "app.h"
+#include "cellrenderer_value.h"
+#include "cellrenderer_timetrack.h"
+#include "workarea.h"
+#include "dialog_color.h"
+#include "eventkey.h"
+
+#include "state_polygon.h"
+#include "state_bline.h"
+#include "state_normal.h"
+#include "state_eyedrop.h"
+#include "state_draw.h"
+
+#include "ducktransform_scale.h"
+#include "ducktransform_translate.h"
+#include "ducktransform_rotate.h"
+
+#include "event_mouse.h"
+#include "event_layerclick.h"
+
+#include "toolbox.h"
+
+#include "dialog_preview.h"
+#include "dialog_soundselect.h"
+
+#include "preview.h"
+#include "audiocontainer.h"
+#include "widget_timeslider.h"
+#include "keyframedial.h"
+
+#include <synfigapp/main.h>
+#include <synfigapp/inputdevice.h>
+
+#include "general.h"
+
+#endif
+
+/* === U S I N G =========================================================== */
+
+using namespace std;
+using namespace etl;
+using namespace synfig;
+using namespace studio;
+using namespace sigc;
+
+/* === M A C R O S ========================================================= */
+
+#define GRAB_HINT_DATA(y) { \
+ String x; \
+ if(synfigapp::Main::settings().get_value(String("pref.")+y+"_hints",x)) \
+ { \
+ set_type_hint((Gdk::WindowTypeHint)atoi(x.c_str())); \
+ } \
+ }
+
+#define DEFAULT_TIME_WINDOW_SIZE (10.0)
+
+#ifndef SMALL_BUTTON
+#define SMALL_BUTTON(button,stockid,tooltip) \
+ button = manage(new class Gtk::Button()); \
+ icon=manage(new Gtk::Image(Gtk::StockID(stockid),iconsize)); \
+ button->add(*icon); \
+ tooltips.set_tip(*button,tooltip); \
+ icon->set_padding(0,0);\
+ icon->show(); \
+ button->set_relief(Gtk::RELIEF_NONE); \
+ button->show()
+#endif
+
+#ifndef NORMAL_BUTTON
+#define NORMAL_BUTTON(button,stockid,tooltip) \
+ button = manage(new class Gtk::Button()); \
+ icon=manage(new Gtk::Image(Gtk::StockID(stockid),Gtk::ICON_SIZE_BUTTON)); \
+ button->add(*icon); \
+ tooltips.set_tip(*button,tooltip); \
+ icon->set_padding(0,0);\
+ icon->show(); \
+ /*button->set_relief(Gtk::RELIEF_NONE);*/ \
+ button->show()
+#endif
+
+#define NEW_SMALL_BUTTON(x,y,z) Gtk::Button *SMALL_BUTTON(x,y,z)
+
+#define NOT_IMPLEMENTED_SLOT sigc::mem_fun(*reinterpret_cast<studio::CanvasViewUIInterface*>(get_ui_interface().get()),&studio::CanvasViewUIInterface::not_implemented)
+
+#define SLOT_EVENT(x) sigc::hide_return(sigc::bind(sigc::mem_fun(*this,&studio::CanvasView::process_event_key),x))
+
+/* === C L A S S E S ======================================================= */
+
+class studio::UniversalScrubber
+{
+ CanvasView *canvas_view;
+
+ bool scrubbing;
+ etl::clock scrub_timer;
+
+ sigc::connection end_scrub_connection;
+public:
+ UniversalScrubber(CanvasView *canvas_view):
+ canvas_view(canvas_view),
+ scrubbing(false)
+ {
+ canvas_view->canvas_interface()->signal_time_changed().connect(
+ sigc::mem_fun(*this,&studio::UniversalScrubber::on_time_changed)
+ );
+ }
+
+ ~UniversalScrubber()
+ {
+ end_scrub_connection.disconnect();
+ }
+
+ void on_time_changed()
+ {
+ // Make sure we are changing the time quickly
+ // before we enable scrubbing
+ if(!scrubbing && scrub_timer()>1)
+ {
+ scrub_timer.reset();
+ return;
+ }
+
+ // If we aren't scrubbing already, enable it
+ if(!scrubbing)
+ {
+ scrubbing=true;
+ audio_container()->start_scrubbing(canvas_view->get_time());
+ }
+
+ // Reset the scrubber ender
+ end_scrub_connection.disconnect();
+ end_scrub_connection=Glib::signal_timeout().connect(
+ sigc::bind_return(
+ sigc::mem_fun(*this,&UniversalScrubber::end_of_scrubbing),
+ false
+ ),
+ 1000
+ );
+
+ // Scrub!
+ audio_container()->scrub(canvas_view->get_time());
+
+ scrub_timer.reset();
+ }
+
+ void end_of_scrubbing()
+ {
+ scrubbing=false;
+ audio_container()->stop_scrubbing();
+ scrub_timer.reset();
+ }
+
+ handle<AudioContainer> audio_container()
+ {
+ assert(canvas_view->audio);
+ return canvas_view->audio;
+ }
+};
+
+class studio::CanvasViewUIInterface : public synfigapp::UIInterface
+{
+ CanvasView *view;
+
+public:
+
+ CanvasViewUIInterface(CanvasView *view):
+ view(view)
+ {
+ view->statusbar->push(_("Idle"));
+ }
+
+ ~CanvasViewUIInterface()
+ {
+ //view->statusbar->pop();
+ //view->progressbar->set_fraction(0);
+ }
+
+ virtual Response confirmation(const std::string &title,
+ const std::string &primaryText,
+ const std::string &secondaryText,
+ const std::string &confirmPhrase,
+ const std::string &cancelPhrase,
+ Response defaultResponse=RESPONSE_OK)
+ {
+ view->present();
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ Gtk::MessageDialog dialog(
+ *view, // Parent
+ primaryText, // Message
+ false, // Markup
+ Gtk::MESSAGE_WARNING, // Type
+ Gtk::BUTTONS_NONE, // Buttons
+ true // Modal
+ );
+
+ if (! title.empty())
+ dialog.set_title(title);
+ if (! secondaryText.empty())
+ dialog.set_secondary_text(secondaryText);
+
+ dialog.add_button(cancelPhrase, RESPONSE_CANCEL);
+ dialog.add_button(confirmPhrase, RESPONSE_OK);
+ dialog.set_default_response(defaultResponse);
+
+ dialog.show_all();
+ return (Response) dialog.run();
+ }
+
+ virtual Response yes_no(const std::string &title, const std::string &message,Response dflt=RESPONSE_YES)
+ {
+ view->present();
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ Gtk::Dialog dialog(
+ title, // Title
+ *view, // Parent
+ true, // Modal
+ true // use_separator
+ );
+ Gtk::Label label(message);
+ label.show();
+
+ dialog.get_vbox()->pack_start(label);
+ dialog.add_button(Gtk::StockID("gtk-yes"),RESPONSE_YES);
+ dialog.add_button(Gtk::StockID("gtk-no"),RESPONSE_NO);
+
+ dialog.set_default_response(dflt);
+ dialog.show();
+ return (Response)dialog.run();
+ }
+ virtual Response yes_no_cancel(const std::string &title, const std::string &message,Response dflt=RESPONSE_YES)
+ {
+ view->present();
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ Gtk::Dialog dialog(
+ title, // Title
+ *view, // Parent
+ true, // Modal
+ true // use_separator
+ );
+ Gtk::Label label(message);
+ label.show();
+
+ dialog.get_vbox()->pack_start(label);
+ dialog.add_button(Gtk::StockID("gtk-yes"),RESPONSE_YES);
+ dialog.add_button(Gtk::StockID("gtk-no"),RESPONSE_NO);
+ dialog.add_button(Gtk::StockID("gtk-cancel"),RESPONSE_CANCEL);
+
+ dialog.set_default_response(dflt);
+ dialog.show();
+ return (Response)dialog.run();
+ }
+ virtual Response ok_cancel(const std::string &title, const std::string &message,Response dflt=RESPONSE_OK)
+ {
+ view->present();
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ Gtk::Dialog dialog(
+ title, // Title
+ *view, // Parent
+ true, // Modal
+ true // use_separator
+ );
+ Gtk::Label label(message);
+ label.show();
+
+ dialog.get_vbox()->pack_start(label);
+ dialog.add_button(Gtk::StockID("gtk-ok"),RESPONSE_OK);
+ dialog.add_button(Gtk::StockID("gtk-cancel"),RESPONSE_CANCEL);
+
+ dialog.set_default_response(dflt);
+ dialog.show();
+ return (Response)dialog.run();
+ }
+
+ virtual bool
+ task(const std::string &task)
+ {
+ if(!view->is_playing_)
+ {
+ view->statusbar->pop();
+ view->statusbar->push(task);
+ }
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ if(view->cancel){return false;}
+ return true;
+ }
+
+ virtual bool
+ error(const std::string &err)
+ {
+ view->statusbar->push(_("ERROR"));
+
+ // If we are in the process of canceling,
+ // then just go ahead and return false --
+ // don't bother displaying a dialog
+ if(view->cancel)return false;
+ Gtk::MessageDialog dialog(*view, err, false, Gtk::MESSAGE_ERROR, Gtk::BUTTONS_CLOSE, true);
+ dialog.show();
+ dialog.run();
+ view->statusbar->pop();
+ return true;
+ }
+
+ virtual bool
+ warning(const std::string &err)
+ {
+ view->statusbar->pop();
+ view->statusbar->push(err);
+
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ if(view->cancel)return false;
+ return true;
+ }
+
+ virtual bool
+ amount_complete(int current, int total)
+ {
+ if(!view->is_playing_)
+ {
+ if(!view->working_depth)
+ {
+ if(current)
+ view->stopbutton->set_sensitive(true);
+ else
+ view->stopbutton->set_sensitive(false);
+ }
+ float x((float)current/(float)total);
+ if(x<0)x=0;
+ else if(x>1)x=1;
+ view->progressbar->set_fraction(x);
+ }
+ //while(studio::App::events_pending())studio::App::iteration(false);
+ if(view->cancel){/*view->cancel=false;*/return false;}
+ return true;
+ }
+
+ void
+ not_implemented()
+ {
+ error(_("Feature not yet implemented"));
+ }
+};
+
+class studio::CanvasViewSelectionManager : public synfigapp::SelectionManager
+{
+ CanvasView *view;
+ CanvasView::LayerTreeModel layer_tree_model;
+ CanvasView::ChildrenTreeModel children_tree_model;
+
+public:
+ CanvasViewSelectionManager(CanvasView *view): view(view) { }
+
+private:
+ void _set_selected_layer(const synfig::Layer::Handle &layer)
+ {
+ view->layer_tree->select_layer(layer);
+/*
+ // Don't change the selection while we are busy
+ // I cannot remember exactly why I put this here...
+ // It musta been for some reason, but I cannot recall.
+ //if(App::Busy::count)
+ // return;
+
+ if(view->layer_tree->get_selection()->get_selected())
+ {
+ const Gtk::TreeRow row = *(view->layer_tree->get_selection()->get_selected());
+
+ // Don't do anything if that layer is already selected
+ if(layer == static_cast<synfig::Layer::Handle>(row[layer_tree_model.layer]))
+ return;
+ }
+ Gtk::TreeModel::Children::iterator iter;
+ if(view->layer_tree_store()->find_layer_row(layer,iter))
+ {
+ Gtk::TreePath path(iter);
+ for(int i=path.get_depth();i;i--)
+ {
+ int j;
+ path=Gtk::TreePath(iter);
+ for(j=i;j;j--)
+ path.up();
+ view->layer_tree->get_tree_view().expand_row(path,false);
+ }
+ view->layer_tree->get_tree_view().scroll_to_row(Gtk::TreePath(iter));
+ view->layer_tree->get_selection()->select(iter);
+ }
+*/
+ }
+public:
+
+ //! Returns the number of layers selected.
+ virtual int get_selected_layer_count()const
+ {
+ return get_selected_layers().size();
+ }
+
+ //! Returns a list of the currently selected layers.
+ virtual LayerList get_selected_layers()const
+ {
+// assert(view->layer_tree);
+
+ if(!view->layer_tree) { synfig::error("%s:%d canvas_view.layer_tree not defined!?", __FILE__, __LINE__); return LayerList(); }
+ return view->layer_tree->get_selected_layers();
+ }
+
+ //! Returns the first layer selected or an empty handle if none are selected.
+ virtual synfig::Layer::Handle get_selected_layer()const
+ {
+// assert(view->layer_tree);
+
+ if(!view->layer_tree) { synfig::error("%s:%d canvas_view.layer_tree not defined!?", __FILE__, __LINE__); return 0; }
+ return view->layer_tree->get_selected_layer();
+ }
+
+ //! Sets which layers should be selected
+ virtual void set_selected_layers(const LayerList &layer_list)
+ {
+// assert(view->layer_tree);
+
+ if(!view->layer_tree) { synfig::error("%s:%d canvas_view.layer_tree not defined!?", __FILE__, __LINE__); return; }
+ view->layer_tree->select_layers(layer_list);
+ //view->get_smach().process_event(EVENT_REFRESH_DUCKS);
+
+ //view->queue_rebuild_ducks();
+ }
+
+ //! Sets which layer should be selected.
+ virtual void set_selected_layer(const synfig::Layer::Handle &layer)
+ {
+// assert(view->layer_tree);
+
+ if(!view->layer_tree) { synfig::error("canvas_view.layer_tree not defined!?"); return; }
+ view->layer_tree->select_layer(layer);
+ //view->queue_rebuild_ducks();
+ }
+
+ //! Clears the layer selection list
+ virtual void clear_selected_layers()
+ {
+ if(!view->layer_tree) return;
+ view->layer_tree->clear_selected_layers();
+ }
+
+ //! Returns the number of value_nodes selected.
+ virtual int get_selected_children_count()const
+ {
+ return get_selected_children().size();
+ }
+
+ static inline void __child_grabber(const Gtk::TreeModel::iterator& iter, ChildrenList* ret)
+ {
+ const CanvasView::ChildrenTreeModel children_tree_model;
+ synfigapp::ValueDesc value_desc((*iter)[children_tree_model.value_desc]);
+ if(value_desc)
+ ret->push_back(value_desc);
+ }
+
+ //! Returns a list of the currently selected value_nodes.
+ virtual ChildrenList get_selected_children()const
+ {
+ if(!view->children_tree) return ChildrenList();
+
+ Glib::RefPtr<Gtk::TreeSelection> selection=view->children_tree->get_selection();
+
+ if(!selection)
+ return ChildrenList();
+
+ ChildrenList ret;
+
+ selection->selected_foreach_iter(
+ sigc::bind(
+ sigc::ptr_fun(
+ &studio::CanvasViewSelectionManager::__child_grabber
+ ),
+ &ret
+ )
+ );
+
+ /*
+ Gtk::TreeModel::Children::iterator iter(view->children_tree_store()->children().begin());
+ iter++;
+ Gtk::TreeModel::Children children = iter->children();
+ for(iter = children.begin(); iter != children.end(); ++iter)
+ {
+ Gtk::TreeModel::Row row = *iter;
+ if(selection->is_selected(row))
+ ret.push_back((synfigapp::ValueDesc)row[children_tree_model.value_desc]);
+ }
+ */
+ return ret;
+ }
+
+ //! Returns the first value_node selected or an empty handle if none are selected.
+ virtual ChildrenList::value_type get_selected_child()const
+ {
+ if(!view->children_tree) return ChildrenList::value_type();
+
+ ChildrenList children(get_selected_children());
+
+ if(children.empty())
+ return ChildrenList::value_type();
+
+ return children.front();
+ }
+
+ //! Sets which value_nodes should be selected
+ virtual void set_selected_children(const ChildrenList &/*children_list*/)
+ {
+ return;
+ }
+
+ //! Sets which value_node should be selected. Empty handle if none.
+ virtual void set_selected_child(const ChildrenList::value_type &/*child*/)
+ {
+ return;
+ }
+
+ //! Clears the value_node selection list
+ virtual void clear_selected_children()
+ {
+ return;
+ }
+
+ int get_selected_layer_parameter_count()const
+ {
+ return get_selected_layer_parameters().size();
+ }
+
+ LayerParamList get_selected_layer_parameters()const
+ {
+ if(!view->layer_tree) return LayerParamList();
+
+ Glib::RefPtr<Gtk::TreeSelection> selection=view->layer_tree->get_selection();
+
+ if(!selection)
+ return LayerParamList();
+
+ LayerParamList ret;
+
+ Gtk::TreeModel::Children children = const_cast<CanvasView*>(view)->layer_tree_store()->children();
+ Gtk::TreeModel::Children::iterator iter;
+ for(iter = children.begin(); iter != children.end(); ++iter)
+ {
+ Gtk::TreeModel::Row row = *iter;
+ Gtk::TreeModel::Children::iterator iter;
+ for(iter=row.children().begin();iter!=row.children().end();iter++)
+ {
+ Gtk::TreeModel::Row row = *iter;
+ if(selection->is_selected(row))
+ ret.push_back(LayerParam(row[layer_tree_model.layer],(Glib::ustring)row[layer_tree_model.id]));
+ }
+ }
+ return ret;
+ }
+
+ LayerParam get_selected_layer_parameter() const
+ {
+ if(!view->layer_tree) return LayerParam();
+ return get_selected_layer_parameters().front();
+ }
+
+ void set_selected_layer_parameters(const LayerParamList &/*layer_param_list*/)
+ {
+ return;
+ }
+
+ void set_selected_layer_param(const LayerParam &/*layer_param*/)
+ {
+ return;
+ }
+
+ void clear_selected_layer_parameters()
+ {
+ return;
+ }
+
+}; // END of class SelectionManager
+
+CanvasView::IsWorking::IsWorking(CanvasView &canvas_view_):
+ canvas_view_(canvas_view_)
+{
+ if(!canvas_view_.working_depth)
+ canvas_view_.stopbutton->set_sensitive(true);
+ canvas_view_.working_depth++;
+ canvas_view_.cancel=false;
+}
+
+CanvasView::IsWorking::~IsWorking()
+{
+ canvas_view_.working_depth--;
+ if(!canvas_view_.working_depth)
+ canvas_view_.stopbutton->set_sensitive(false);
+}
+
+CanvasView::IsWorking::operator bool()const
+{
+ if(canvas_view_.cancel)
+ return false;
+ return true;
+}
+
+/* === M E T H O D S ======================================================= */
+
+CanvasView::CanvasView(etl::loose_handle<Instance> instance,etl::handle<synfigapp::CanvasInterface> canvas_interface_):
+ smach_ (this),
+ instance_ (instance),
+ canvas_interface_ (canvas_interface_),
+ //layer_tree_store_ (LayerTreeStore::create(canvas_interface_)),
+ //children_tree_store_ (ChildrenTreeStore::create(canvas_interface_)),
+ //keyframe_tree_store_ (KeyframeTreeStore::create(canvas_interface_)),
+ time_adjustment_ (0,0,25,0,0,0),
+ time_window_adjustment_ (0,0,25,0,0,0),
+ quality_adjustment_ (8,1,10,1,1,0),
+ future_onion_adjustment_ (0,0,2,1,1,0),
+ past_onion_adjustment_ (0,0,2,1,1,0),
+ statusbar (manage(new class Gtk::Statusbar())),
+
+ timeslider (new Widget_Timeslider),
+ widget_kf_list (new Widget_Keyframe_List),
+
+ ui_interface_ (new CanvasViewUIInterface(this)),
+ selection_manager_ (new CanvasViewSelectionManager(this)),
+ is_playing_ (false),
+
+ working_depth (0),
+ cancel (false),
+
+ canvas_properties (*this,canvas_interface_),
+ canvas_options (this),
+ render_settings (*this,canvas_interface_),
+ waypoint_dialog (*this,canvas_interface_->get_canvas()),
+ keyframe_dialog (*this,canvas_interface_),
+ preview_dialog (new Dialog_Preview),
+ sound_dialog (new Dialog_SoundSelect(*this,canvas_interface_))
+{
+ layer_tree=0;
+ children_tree=0;
+ duck_refresh_flag=true;
+ toggling_ducks_=false;
+ changing_resolution_=false;
+ updating_quality_=false;
+ toggling_show_grid=false;
+ toggling_snap_grid=false;
+ toggling_onion_skin=false;
+
+ smach_.set_default_state(&state_normal);
+
+ disp_audio = new Widget_Sound();
+
+ //synfig::info("Canvasview: Entered constructor");
+ // Minor hack
+ get_canvas()->set_time(0);
+ //layer_tree_store_->rebuild();
+
+ // Set up the UI and Selection managers
+ canvas_interface()->set_ui_interface(get_ui_interface());
+ canvas_interface()->set_selection_manager(get_selection_manager());
+ rebuild_ducks_queued=false;
+
+ //notebook=manage(new class Gtk::Notebook());
+ //Gtk::VPaned *vpaned = manage(new class Gtk::VPaned());
+ //vpaned->pack1(*create_work_area(), Gtk::EXPAND|Gtk::SHRINK);
+ //vpaned->pack2(*notebook, Gtk::SHRINK);
+ //vpaned->show_all();
+
+ //notebook->show();
+
+ //notebook->append_page(*create_layer_tree(),_("Layers"));
+ //notebook->append_page(*create_children_tree(),_("Children"));
+ //notebook->append_page(*create_keyframe_tree(),_("Keyframes"));
+
+ //synfig::info("Canvasview: Before big chunk of allocation and tabling stuff");
+ //create all allocated stuff for this canvas
+ audio = new AudioContainer();
+
+ Gtk::Table *layout_table= manage(new class Gtk::Table(1, 4, false));
+ //layout_table->attach(*vpaned, 0, 1, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
+ layout_table->attach(*create_work_area(), 0, 1, 1, 2, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
+ layout_table->attach(*create_display_bar(), 0, 1, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK|Gtk::FILL, 0, 0);
+ init_menus();
+ //layout_table->attach(*App::ui_manager()->get_widget("/menu-main"), 0, 1, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK|Gtk::FILL, 0, 0);
+
+ layout_table->attach(*create_time_bar(), 0, 1, 3, 4, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK|Gtk::FILL, 0, 0);
+ layout_table->attach(*create_status_bar(), 0, 1, 4, 5, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK|Gtk::FILL, 0, 0);
+
+ update_title();
+
+ layout_table->show();
+ add(*layout_table);
+
+ //set_transient_for(*App::toolbox);
+
+ //synfig::info("Canvasview: Before Signals");
+ /*
+ -- ** -- Signals -------------------------------------------------------------
+ */
+
+ canvas_interface()->signal_dirty_preview().connect(sigc::mem_fun(*this,&studio::CanvasView::on_dirty_preview));
+ canvas_interface()->signal_mode_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::on_mode_changed));
+
+ canvas_interface()->signal_time_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::on_time_changed));
+
+ //canvas_interface()->signal_time_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::refresh_tables));
+ canvas_interface()->signal_id_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::on_id_changed));
+ canvas_interface()->signal_rend_desc_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::refresh_rend_desc));
+ waypoint_dialog.signal_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::on_waypoint_changed));
+ waypoint_dialog.signal_delete().connect(sigc::mem_fun(*this,&studio::CanvasView::on_waypoint_delete));
+
+ //MODIFIED TIME ADJUSTMENT STUFF....
+ time_window_adjustment().set_child_adjustment(&time_adjustment());
+ time_window_adjustment().signal_value_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::refresh_time_window));
+ time_adjustment().signal_value_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::time_was_changed));
+
+ work_area->signal_layer_selected().connect(sigc::mem_fun(*this,&studio::CanvasView::workarea_layer_selected));
+ work_area->signal_input_device_changed().connect(sigc::mem_fun(*this,&studio::CanvasView::on_input_device_changed));
+
+ canvas_interface()->signal_canvas_added().connect(
+ sigc::hide(
+ sigc::mem_fun(*instance,&studio::Instance::refresh_canvas_tree)
+ )
+ );
+ canvas_interface()->signal_canvas_removed().connect(
+ sigc::hide(
+ sigc::mem_fun(*instance,&studio::Instance::refresh_canvas_tree)
+ )
+ );
+
+ canvas_interface()->signal_layer_param_changed().connect(
+ sigc::hide(
+ sigc::hide(
+ SLOT_EVENT(EVENT_REFRESH_DUCKS)
+ )
+ )
+ );
+
+ //MUCH TIME STUFF TAKES PLACE IN HERE
+ refresh_rend_desc();
+ refresh_time_window();
+
+ /*! \todo We shouldn't need to do this at construction --
+ ** This should be performed at the first time the window
+ ** becomes visible.
+ */
+ work_area->queue_render_preview();
+
+ // If the canvas is really big, zoom out so that we can fit it all in the window
+ /*! \todo In other words, this is a zoom-to-fit, and should be
+ ** in its own function.
+ */
+ int w=get_canvas()->rend_desc().get_w()+70;
+ int h=get_canvas()->rend_desc().get_h()+70;
+ while(w>700 || h>600)
+ {
+ // Minor hack:
+ // zoom_out() =>
+ // WorkArea::async_update_preview() =>
+ // WorkArea::set_zoom(float) =>
+ // WorkArea::async_update_preview() =>
+ // desc.set_time(cur_time), where cur_time isn't initialized
+ work_area->set_time(0);
+ work_area->zoom_out();
+ w=round_to_int(get_canvas()->rend_desc().get_w()*work_area->get_zoom()+70);
+ h=round_to_int(get_canvas()->rend_desc().get_h()*work_area->get_zoom()+70);
+ }
+ if(w>700)w=700;
+ if(h>600)h=600;
+ set_default_size(w,h);
+ property_window_position().set_value(Gtk::WIN_POS_NONE);
+
+ std::list<Gtk::TargetEntry> listTargets;
+ listTargets.push_back( Gtk::TargetEntry("STRING") );
+ listTargets.push_back( Gtk::TargetEntry("text/plain") );
+ listTargets.push_back( Gtk::TargetEntry("image") );
+
+ drag_dest_set(listTargets);
+ signal_drag_data_received().connect( sigc::mem_fun(*this, &studio::CanvasView::on_drop_drag_data_received) );
+
+ /*
+ Time length(get_canvas()->rend_desc().get_time_end()-get_canvas()->rend_desc().get_time_start());
+ if(length<10.0)
+ {
+ time_window_adjustment().set_page_increment(length);
+ time_window_adjustment().set_page_size(length);
+ }
+ else
+ {
+ time_window_adjustment().set_page_increment(10.0);
+ time_window_adjustment().set_page_size(10.0);
+ }
+ */
+
+ //synfig::info("Canvasview: Before Sound Hookup");
+ //load sound info from meta data
+ {
+ //synfig::warning("Should load Audio: %s with %s offset",apath.c_str(),aoffset.c_str());
+
+ on_audio_file_notify(); //redundant setting of the metadata, but oh well, it's no big deal :)
+ on_audio_offset_notify();
+
+ //signal connection - since they are all associated with the canvas view
+
+ //hook in signals for sound options box
+ sound_dialog->signal_file_changed().connect(sigc::mem_fun(*this,&CanvasView::on_audio_file_change));
+ sound_dialog->signal_offset_changed().connect(sigc::mem_fun(*this,&CanvasView::on_audio_offset_change));
+
+ //attach to the preview when it's visible
+ //preview_dialog->get_widget().signal_play().connect(sigc::mem_fun(*this,&CanvasView::play_audio));
+ //preview_dialog->get_widget().signal_stop().connect(sigc::mem_fun(*this,&CanvasView::stop_audio));
+
+ //hook to metadata signals
+ get_canvas()->signal_meta_data_changed("audiofile").connect(sigc::mem_fun(*this,&CanvasView::on_audio_file_notify));
+ get_canvas()->signal_meta_data_changed("audiooffset").connect(sigc::mem_fun(*this,&CanvasView::on_audio_offset_notify));
+
+ //universal_scrubber=std::auto_ptr<UniversalScrubber>(new UniversalScrubber(this));
+ }
+
+ //synfig::info("Canvasview: Before Final time set up");
+ //MORE TIME STUFF
+ time_window_adjustment().set_value(get_canvas()->rend_desc().get_time_start());
+ time_window_adjustment().value_changed();
+
+ GRAB_HINT_DATA("canvas_view");
+ /*
+ {
+ set_skip_taskbar_hint(true);
+ set_skip_pager_hint(true);
+ set_type_hint(Gdk::WINDOW_TYPE_HINT_UTILITY);
+ }
+ */
+
+ refresh_rend_desc();
+ hide_tables();
+
+ on_time_changed();
+ //synfig::info("Canvasview: Constructor Done");
+}
+
+CanvasView::~CanvasView()
+{
+ signal_deleted()();
+
+ App::ui_manager()->remove_action_group(action_group);
+
+ // Shut down the smach
+ smach_.egress();
+ smach_.set_default_state(0);
+
+ // We want to ensure that the UI_Manager and
+ // the selection manager get destructed right now.
+ ui_interface_.reset();
+ selection_manager_.reset();
+
+ // Delete any external widgets
+ for(;!ext_widget_book_.empty();ext_widget_book_.erase(ext_widget_book_.begin()))
+ {
+ if(ext_widget_book_.begin()->second)
+ delete ext_widget_book_.begin()->second;
+ }
+
+ //delete preview
+ audio.reset();
+
+ hide();
+
+ // don't be calling on_dirty_preview once this object has been deleted;
+ // this was causing a crash before
+ canvas_interface()->signal_dirty_preview().clear();
+
+ if (getenv("SYNFIG_DEBUG_DESTRUCTORS"))
+ synfig::info("CanvasView::~CanvasView(): Deleted");
+}
+
+std::list<int>&
+CanvasView::get_pixel_sizes()
+{
+ // prime factors of 120 are 2, 2, 2, 3, 5 - see TILE_SIZE in synfig-core/trunk/src/synfig/target_tile.h
+ static int pixel_size_array[] = {2,3,4,5,6,8,10,12,15,20,24,30,40,60,120};
+ static list<int> pixel_sizes = list<int>(pixel_size_array, pixel_size_array + sizeof(pixel_size_array) / sizeof(int));
+
+ return pixel_sizes;
+}
+
+Gtk::Widget *
+CanvasView::create_time_bar()
+{
+ Gtk::Image *icon;
+
+ //Setup the Time Slider and the Time window scroll
+ Gtk::HScrollbar *time_window_scroll = manage(new class Gtk::HScrollbar(time_window_adjustment()));
+ //Gtk::HScrollbar *time_scroll = manage(new class Gtk::HScrollbar(time_adjustment()));
+ //TIME BAR TEMPORARY POSITION
+ //Widget_Timeslider *time_scroll = manage(new Widget_Timeslider);
+ timeslider->set_time_adjustment(&time_adjustment());
+ timeslider->set_bounds_adjustment(&time_window_adjustment());
+ //layout_table->attach(*timeslider, 0, 1, 2, 3, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK|Gtk::FILL);
+ //Setup the keyframe list widget
+ widget_kf_list->set_time_adjustment(&time_adjustment());
+ widget_kf_list->set_canvas_interface(canvas_interface());
+ widget_kf_list->show();
+
+ tooltips.set_tip(*time_window_scroll,_("Moves the time window"));
+ tooltips.set_tip(*timeslider,_("Changes the current time"));
+ time_window_scroll->show();
+ timeslider->show();
+ //time_window_scroll->set_flags(Gtk::CAN_FOCUS); // Uncomment this produce bad render of the HScroll
+ timeslider->set_flags(Gtk::CAN_FOCUS);
+
+ //time_scroll->signal_value_changed().connect(sigc::mem_fun(*work_area, &studio::WorkArea::render_preview_hook));
+ //time_scroll->set_update_policy(Gtk::UPDATE_DISCONTINUOUS);
+
+ //Setup the Animation Mode Button and the Keyframe Lock button
+ Gtk::IconSize iconsize=Gtk::IconSize::from_name("synfig-small_icon");
+ SMALL_BUTTON(animatebutton,"gtk-yes",_("Animate"));
+ animatebutton->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::on_animate_button_pressed));
+ animatebutton->show();
+
+ //Setup the audio display
+ disp_audio->set_size_request(-1,32); //disp_audio->show();
+ disp_audio->set_time_adjustment(&time_adjustment());
+ disp_audio->signal_start_scrubbing().connect(
+ sigc::mem_fun(*audio,&AudioContainer::start_scrubbing)
+ );
+ disp_audio->signal_scrub().connect(
+ sigc::mem_fun(*audio,&AudioContainer::scrub)
+ );
+ disp_audio->signal_stop_scrubbing().connect(
+ sigc::mem_fun(*audio,&AudioContainer::stop_scrubbing)
+ );
+ //Setup the current time widget
+ current_time_widget=manage(new Widget_Time);
+ current_time_widget->set_value(get_time());
+ current_time_widget->set_fps(get_canvas()->rend_desc().get_frame_rate());
+ current_time_widget->signal_value_changed().connect(
+ sigc::mem_fun(*this,&CanvasView::on_current_time_widget_changed)
+ );
+ current_time_widget->set_size_request(0,-1); // request horizontal shrink
+ tooltips.set_tip(*current_time_widget,_("Current time"));
+ current_time_widget->show();
+
+ //Setup the FrameDial widget
+ framedial = manage(new class FrameDial());
+ framedial->signal_seek_begin().connect(
+ sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time), Time::begin())
+ );
+ framedial->signal_seek_prev_frame().connect(
+ sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_frame), -1)
+ );
+ framedial->signal_play_stop().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::on_play_stop_pressed)
+ );
+ framedial->signal_seek_next_frame().connect(
+ sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_frame), 1)
+ );
+ framedial->signal_seek_end().connect(
+ sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time), Time::end())
+ );
+ framedial->show();
+
+ //Setup the KeyFrameDial widget
+ KeyFrameDial *keyframedial = Gtk::manage(new class KeyFrameDial());
+ keyframedial->signal_seek_prev_keyframe().connect(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_prev_keyframe));
+ keyframedial->signal_seek_next_keyframe().connect(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_next_keyframe));
+ keyframedial->signal_lock_keyframe().connect(sigc::mem_fun(*this, &studio::CanvasView::on_keyframe_button_pressed));
+ keyframedial->show();
+ keyframebutton=keyframedial->get_lock_button();
+
+ timebar = Gtk::manage(new class Gtk::Table(5, 4, false));
+
+ //Adjust both widgets to be the same as the
+ int header_height = 0;
+ if(getenv("SYNFIG_TIMETRACK_HEADER_HEIGHT"))
+ header_height = atoi(getenv("SYNFIG_TIMETRACK_HEADER_HEIGHT"));
+ if (header_height < 3)
+ header_height = 24;
+ timeslider->set_size_request(-1,header_height-header_height/3+1);
+ widget_kf_list->set_size_request(-1,header_height/3+1);
+
+ //Attach widgets to the timebar
+ //timebar->attach(*manage(disp_audio), 1, 5, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK);
+ timebar->attach(*current_time_widget, 0, 1, 0, 2, Gtk::SHRINK|Gtk::FILL, Gtk::SHRINK|Gtk::FILL, 0, 0);
+ timebar->attach(*framedial, 0, 1, 2, 3, Gtk::SHRINK, Gtk::SHRINK);
+ timebar->attach(*timeslider, 1, 3, 1, 2, Gtk::FILL|Gtk::SHRINK, Gtk::FILL|Gtk::SHRINK);
+ timebar->attach(*widget_kf_list, 1, 3, 0, 1, Gtk::FILL|Gtk::EXPAND, Gtk::FILL|Gtk::SHRINK);
+ timebar->attach(*time_window_scroll, 1, 3, 2, 3, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK);
+ timebar->attach(*keyframedial, 3, 4, 0, 2, Gtk::SHRINK, Gtk::SHRINK);
+ timebar->attach(*animatebutton, 4, 5, 0, 2, Gtk::SHRINK, Gtk::SHRINK);
+ //timebar->attach(*keyframebutton, 1, 2, 3, 4, Gtk::SHRINK, Gtk::SHRINK);
+
+ timebar->show();
+
+ return timebar;
+}
+
+Gtk::Widget *
+CanvasView::create_work_area()
+{
+ work_area=std::auto_ptr<WorkArea>(new class studio::WorkArea(canvas_interface_));
+ work_area->set_instance(get_instance());
+ work_area->set_canvas(get_canvas());
+ work_area->set_canvas_view(this);
+ work_area->set_progress_callback(get_ui_interface().get());
+ work_area->signal_popup_menu().connect(sigc::mem_fun(*this, &studio::CanvasView::popup_main_menu));
+ work_area->show();
+ return work_area.get();
+}
+
+Gtk::Widget*
+CanvasView::create_status_bar()
+{
+ Gtk::Image *icon;
+ Gtk::IconSize iconsize=Gtk::IconSize::from_name("synfig-small_icon");
+ cancel=false;
+
+ // Create the status bar at the bottom of the window
+ Gtk::Table *statusbartable= manage(new class Gtk::Table(5, 1, false));
+// statusbar = manage(new class Gtk::Statusbar()); // This is already done at construction
+ progressbar =manage(new class Gtk::ProgressBar());
+ SMALL_BUTTON(stopbutton,"gtk-stop",_("Stop"));
+ SMALL_BUTTON(refreshbutton,"gtk-refresh",_("Refresh"));
+ //SMALL_BUTTON(treetogglebutton,"gtk-go-down",_("Toggle Layer Tree"));
+// NEW_SMALL_BUTTON(raisebutton,"gtk-go-up",_("Raise Layer"));
+// NEW_SMALL_BUTTON(lowerbutton,"gtk-go-down",_("Lower Layer"));
+ //statusbartable->attach(*treetogglebutton, 0, 1, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0);
+// statusbartable->attach(*lowerbutton, 0, 1, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0);
+// statusbartable->attach(*raisebutton, 1, 2, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0);
+
+ statusbartable->attach(*statusbar, 1, 2, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
+ statusbartable->attach(*progressbar, 2, 3, 0, 1, Gtk::SHRINK, Gtk::EXPAND|Gtk::FILL, 0, 0);
+ statusbartable->attach(*refreshbutton, 3, 4, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0);
+ statusbartable->attach(*stopbutton, 4, 5, 0, 1, Gtk::SHRINK, Gtk::SHRINK, 0, 0);
+ statusbar->set_has_resize_grip(false);
+ statusbar->show();
+ stopbutton->show();
+ refreshbutton->show();
+ progressbar->show();
+ stopbutton->set_sensitive(false);
+
+ //refreshbutton->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::on_refresh_pressed));
+ //stopbutton->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::stop));
+ //treetogglebutton->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::toggle_tables));
+
+ refreshbutton->signal_clicked().connect(SLOT_EVENT(EVENT_REFRESH));
+ stopbutton->signal_clicked().connect(SLOT_EVENT(EVENT_STOP));
+
+ statusbartable->show_all();
+ return statusbartable;
+}
+
+Gtk::Widget*
+CanvasView::create_display_bar()
+{
+ displaybar = manage(new class Gtk::Table(1, 7, false));
+
+ // Setup the ToggleDuckDial widget
+ toggleducksdial = Gtk::manage(new class ToggleDucksDial());
+
+ Duck::Type m = work_area->get_type_mask();
+ toggleducksdial->update_toggles(m);
+
+ toggleducksdial->signal_ducks_position().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_POSITION)
+ );
+ toggleducksdial->signal_ducks_vertex().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_VERTEX)
+ );
+ toggleducksdial->signal_ducks_tangent().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_TANGENT)
+ );
+ toggleducksdial->signal_ducks_radius().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_RADIUS)
+ );
+ toggleducksdial->signal_ducks_width().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_WIDTH)
+ );
+ toggleducksdial->signal_ducks_angle().connect(
+ sigc::bind(sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),Duck::TYPE_ANGLE)
+ );
+ toggleducksdial->show();
+
+ // Set up the ResolutionDial widget
+ resolutiondial=Gtk::manage(new class ResolutionDial());
+
+ resolutiondial->update_lowres(work_area->get_low_resolution_flag());
+ resolutiondial->signal_increase_resolution().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::decrease_low_res_pixel_size));
+ resolutiondial->signal_decrease_resolution().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::increase_low_res_pixel_size));
+ resolutiondial->signal_use_low_resolution().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::toggle_low_res_pixel_flag));
+ resolutiondial->show();
+
+ // Set up a separator
+ Gtk::VSeparator *separator1 = Gtk::manage(new class Gtk::VSeparator());
+ separator1->show();
+ Gtk::VSeparator *separator2 = Gtk::manage(new class Gtk::VSeparator());
+ separator2->show();
+
+ // Set up quality spin button
+ quality_spin=Gtk::manage(new class Gtk::SpinButton(quality_adjustment_));
+ quality_spin->signal_value_changed().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::update_quality));
+ tooltips.set_tip(*quality_spin, _("Quality (lower is better)"));
+ quality_spin->show();
+
+ // Set up the show grid toggle button
+ show_grid = Gtk::manage(new class Gtk::ToggleButton());
+ show_grid->set_active(work_area->grid_status());
+ Gtk::Image *icon = manage(new Gtk::Image(Gtk::StockID("synfig-toggle_show_grid"), Gtk::IconSize::from_name("synfig-small_icon")));
+ icon->set_padding(0, 0);
+ icon->show();
+ show_grid->add(*icon);
+ show_grid->signal_toggled().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::toggle_show_grid));
+ tooltips.set_tip(*show_grid, _("Show grid when enabled"));
+ show_grid->set_relief(Gtk::RELIEF_NONE);
+ show_grid->show();
+
+ // Set up the snap to grid toggle button
+ snap_grid = Gtk::manage(new class Gtk::ToggleButton());
+ snap_grid->set_active(work_area->grid_status());
+ Gtk::Image *icon2 = manage(new Gtk::Image(Gtk::StockID("synfig-toggle_snap_grid"), Gtk::IconSize::from_name("synfig-small_icon")));
+ icon2->set_padding(0, 0);
+ icon2->show();
+ snap_grid->add(*icon2);
+ snap_grid->signal_toggled().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::toggle_snap_grid));
+ tooltips.set_tip(*snap_grid, _("Snap grid when enabled"));
+ snap_grid->set_relief(Gtk::RELIEF_NONE);
+ snap_grid->show();
+
+ // Set up the onion skin toggle button
+ onion_skin = Gtk::manage(new class Gtk::ToggleButton());
+ onion_skin->set_active(work_area->get_onion_skin());
+ Gtk::Image *icon3 = manage(new Gtk::Image(Gtk::StockID("synfig-toggle_onion_skin"), Gtk::IconSize::from_name("synfig-small_icon")));
+ icon3->set_padding(0, 0);
+ icon3->show();
+ onion_skin->add(*icon3);
+ onion_skin->signal_toggled().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::toggle_onion_skin));
+ tooltips.set_tip(*onion_skin, _("Shows onion skin when enabled"));
+ onion_skin->set_relief(Gtk::RELIEF_NONE);
+ onion_skin->show();
+
+ // Set up past onion skin spin button
+ past_onion_spin=Gtk::manage(new class Gtk::SpinButton(past_onion_adjustment_));
+ past_onion_spin->signal_value_changed().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::set_onion_skins));
+ tooltips.set_tip(*past_onion_spin, _("Past onion skins"));
+ past_onion_spin->show();
+
+ // Set up future onion skin spin button
+ future_onion_spin=Gtk::manage(new class Gtk::SpinButton(future_onion_adjustment_));
+ future_onion_spin->signal_value_changed().connect(
+ sigc::mem_fun(*this, &studio::CanvasView::set_onion_skins));
+ tooltips.set_tip(*future_onion_spin, _("Future onion skins"));
+ future_onion_spin->show();
+
+
+ displaybar->attach(*toggleducksdial, 0, 1, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*separator1, 1, 2, 0, 1, Gtk::FILL, Gtk::FILL);
+ displaybar->attach(*resolutiondial, 2, 3, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*separator2, 3, 4, 0, 1, Gtk::FILL, Gtk::FILL);
+ displaybar->attach(*quality_spin, 4, 5, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*show_grid, 5, 6, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*snap_grid, 6, 7, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*past_onion_spin, 7, 8, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*onion_skin, 8, 9, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+ displaybar->attach(*future_onion_spin, 9, 10, 0, 1, Gtk::SHRINK, Gtk::SHRINK);
+
+ displaybar->show();
+
+ return displaybar;
+
+}
+
+void
+CanvasView::on_current_time_widget_changed()
+{
+ set_time(current_time_widget->get_value());
+
+ // show the value being used - it will have been rounded to nearest frame
+ // this was already being done elsewhere, but only if the time was really changed;
+ // if the current time was 6f and the user edited it to 6.1f, then the 6.1f would
+ // be left in the display without the following line to fix it
+ current_time_widget->set_value(get_time());
+ current_time_widget->set_position(-1); // leave the cursor at the end
+}
+
+// Gtk::Widget*
+// CanvasView::create_children_tree()
+// {
+// // Create the layer tree
+// children_tree=manage(new class ChildrenTree());
+//
+// // Set up the layer tree
+// //children_tree->set_model(children_tree_store());
+// if(children_tree)children_tree->set_time_adjustment(time_adjustment());
+// if(children_tree)children_tree->show();
+//
+// // Connect Signals
+// if(children_tree)children_tree->signal_edited_value().connect(sigc::mem_fun(*this, &studio::CanvasView::on_edited_value));
+// if(children_tree)children_tree->signal_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_children_user_click));
+// if(children_tree)children_tree->signal_waypoint_clicked_childrentree().connect(sigc::mem_fun(*this, &studio::CanvasView::on_waypoint_clicked_canvasview));
+// if(children_tree)children_tree->get_selection()->signal_changed().connect(SLOT_EVENT(EVENT_REFRESH_DUCKS));
+//
+// return children_tree;
+// }
+
+// Gtk::Widget*
+// CanvasView::create_keyframe_tree()
+// {
+// keyframe_tree=manage(new KeyframeTree());
+//
+// //keyframe_tree->get_selection()->set_mode(Gtk::SELECTION_MULTIPLE);
+// //keyframe_tree->show();
+// //keyframe_tree->set_model(keyframe_tree_store());
+// keyframe_tree->set_editable(true);
+// //keyframe_tree->signal_edited().connect(sigc::hide_return(sigc::mem_fun(*canvas_interface(), &synfigapp::CanvasInterface::update_keyframe)));
+//
+// keyframe_tree->signal_event().connect(sigc::mem_fun(*this, &studio::CanvasView::on_keyframe_tree_event));
+//
+// Gtk::ScrolledWindow *scroll_layer_tree = manage(new class Gtk::ScrolledWindow());
+// scroll_layer_tree->set_flags(Gtk::CAN_FOCUS);
+// scroll_layer_tree->set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
+// scroll_layer_tree->add(*keyframe_tree);
+// scroll_layer_tree->set_shadow_type(Gtk::SHADOW_ETCHED_IN);
+// //scroll_layer_tree->show();
+//
+//
+// Gtk::Table *layout_table= manage(new Gtk::Table(1, 2, false));
+// layout_table->attach(*scroll_layer_tree, 0, 1, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
+//
+// Gtk::Image *icon;
+// Gtk::IconSize iconsize(Gtk::IconSize::from_name("synfig-small_icon"));
+//
+// NEW_SMALL_BUTTON(button_add,"gtk-add",_("New Keyframe"));
+// NEW_SMALL_BUTTON(button_duplicate,"synfig-duplicate",_("Duplicate Keyframe"));
+// NEW_SMALL_BUTTON(button_delete,"gtk-delete",_("Delete Keyframe"));
+//
+// Gtk::HBox *hbox(manage(new Gtk::HBox()));
+// layout_table->attach(*hbox, 0, 1, 1, 2, Gtk::EXPAND|Gtk::FILL, Gtk::SHRINK, 0, 0);
+//
+// hbox->pack_start(*button_add,Gtk::PACK_SHRINK);
+// hbox->pack_start(*button_duplicate,Gtk::PACK_SHRINK);
+// hbox->pack_start(*button_delete,Gtk::PACK_SHRINK);
+//
+// /*
+// button_raise->set_relief(Gtk::RELIEF_HALF);
+// button_lower->set_relief(Gtk::RELIEF_HALF);
+// button_duplicate->set_relief(Gtk::RELIEF_HALF);
+// button_delete->set_relief(Gtk::RELIEF_HALF);
+// */
+//
+// button_add->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::on_keyframe_add_pressed));
+// button_duplicate->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::on_keyframe_duplicate_pressed));
+// button_delete->signal_clicked().connect(sigc::mem_fun(*this, &studio::CanvasView::on_keyframe_remove_pressed));
+//
+// //layout_table->show_all();
+//
+// keyframe_tab_child=layout_table;
+//
+//
+// layout_table->hide();
+//
+// return layout_table;
+// }
+
+// Gtk::Widget*
+// CanvasView::create_layer_tree()
+// {
+// // Create the layer tree
+// printf("CanvasView::create_layer_tree()\n");
+// layer_tree=manage(new class LayerTree());
+//
+// // Set up the layer tree
+// //layer_tree->set_model(layer_tree_store());
+// layer_tree->set_time_adjustment(time_adjustment());
+// layer_tree->show();
+//
+// // Connect Signals
+// layer_tree->signal_layer_toggle().connect(sigc::mem_fun(*this, &studio::CanvasView::on_layer_toggle));
+// layer_tree->signal_edited_value().connect(sigc::mem_fun(*this, &studio::CanvasView::on_edited_value));
+// layer_tree->signal_layer_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_layer_user_click));
+// layer_tree->signal_param_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_children_user_click));
+// layer_tree->signal_waypoint_clicked_layertree().connect(sigc::mem_fun(*this, &studio::CanvasView::on_waypoint_clicked_canvasview));
+// layer_tree->get_selection()->signal_changed().connect(SLOT_EVENT(EVENT_REFRESH_DUCKS));
+//
+// layer_tree->hide();
+// return layer_tree;
+// }
+
+void
+CanvasView::init_menus()
+{
+/*
+ mainmenu.set_accel_group(get_accel_group());
+ mainmenu.set_accel_path("<Canvas-view>");
+
+ filemenu.set_accel_group(get_accel_group());
+ filemenu.set_accel_path("<Canvas-view>/File");
+
+ editmenu.set_accel_group(get_accel_group());
+ editmenu.set_accel_path("<Canvas-view>/Edit");
+
+ layermenu.set_accel_group(get_accel_group());
+ layermenu.set_accel_path("<Canvas-view>/Layer");
+*/
+ //cache the position of desired widgets
+
+ /*Menus to worry about:
+ - filemenu
+ - editmenu
+ - layermenu
+ - duckmaskmenu
+ - mainmenu
+ - canvasmenu
+ - viewmenu
+ */
+ action_group = Gtk::ActionGroup::create();
+
+ //action_group->add( Gtk::Action::create("MenuFile", _("_File")) );
+ action_group->add( Gtk::Action::create("new", Gtk::Stock::NEW),
+ sigc::hide_return(sigc::ptr_fun(&studio::App::new_instance))
+ );
+ action_group->add( Gtk::Action::create("open", Gtk::Stock::OPEN),
+ sigc::hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::open))
+ );
+ action_group->add( Gtk::Action::create("save", Gtk::Stock::SAVE),
+ hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::save))
+ );
+ action_group->add( Gtk::Action::create("save-as", Gtk::Stock::SAVE_AS),
+ sigc::hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::dialog_save_as))
+ );
+ action_group->add( Gtk::Action::create("revert", Gtk::Stock::REVERT_TO_SAVED),
+ sigc::hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::safe_revert))
+ );
+ action_group->add( Gtk::Action::create("cvs-add", Gtk::StockID("synfig-cvs_add")),
+ sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_add))
+ );
+ action_group->add( Gtk::Action::create("cvs-update", Gtk::StockID("synfig-cvs_update")),
+ sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_update))
+ );
+ action_group->add( Gtk::Action::create("cvs-revert", Gtk::StockID("synfig-cvs_revert")),
+ sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_revert))
+ );
+ action_group->add( Gtk::Action::create("cvs-commit", Gtk::StockID("synfig-cvs_commit")),
+ sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_commit))
+ );
+ action_group->add( Gtk::Action::create("import", _("Import")),
+ sigc::hide_return(sigc::mem_fun(*this, &studio::CanvasView::image_import))
+ );
+ action_group->add( Gtk::Action::create("render", _("Render")),
+ sigc::mem_fun0(render_settings,&studio::RenderSettings::present)
+ );
+ action_group->add( Gtk::Action::create("preview", _("Preview")),
+ sigc::mem_fun(*this,&CanvasView::on_preview_option)
+ );
+ action_group->add( Gtk::Action::create("sound", _("Sound File")),
+ sigc::mem_fun(*this,&CanvasView::on_audio_option)
+ );
+ action_group->add( Gtk::Action::create("options", _("Options")),
+ sigc::mem_fun0(canvas_options,&studio::CanvasOptions::present)
+ );
+ action_group->add( Gtk::Action::create("close", Gtk::StockID("gtk-close"), _("Close Window")),
+ sigc::hide_return(sigc::mem_fun(*this,&studio::CanvasView::close_view))
+ );
+ action_group->add( Gtk::Action::create("close-document", Gtk::StockID("gtk-close"), _("Close Document")),
+ sigc::hide_return(sigc::mem_fun(*this,&studio::CanvasView::close_instance))
+ );
+ action_group->add( Gtk::Action::create("quit", Gtk::StockID("gtk-quit"), _("Quit")),
+ sigc::hide_return(sigc::ptr_fun(&studio::App::quit))
+ );
+
+ //action_group->add( Gtk::Action::create("undo", Gtk::StockID("gtk-undo")),
+ // SLOT_EVENT(EVENT_UNDO)
+ //);
+
+ //action_group->add( Gtk::Action::create("redo", Gtk::StockID("gtk-redo")),
+ // SLOT_EVENT(EVENT_REDO)
+ //);
+
+ action_group->add( Gtk::Action::create("select-all-ducks", _("Select All Ducks")),
+ sigc::mem_fun(*work_area,&studio::WorkArea::select_all_ducks)
+ );
+
+ action_group->add( Gtk::Action::create("unselect-all-ducks", _("Unselect All Ducks")),
+ sigc::mem_fun(*work_area,&studio::WorkArea::unselect_all_ducks)
+ );
+
+ action_group->add( Gtk::Action::create("select-all-layers", _("Select All Layers")),
+ sigc::mem_fun(*this,&CanvasView::on_select_layers)
+ );
+
+ action_group->add( Gtk::Action::create("unselect-all-layers", _("Unselect All Layers")),
+ sigc::mem_fun(*this,&CanvasView::on_unselect_layers)
+ );
+
+ action_group->add( Gtk::Action::create("stop", Gtk::StockID("gtk-stop")),
+ SLOT_EVENT(EVENT_STOP)
+ );
+
+ action_group->add( Gtk::Action::create("refresh", Gtk::StockID("gtk-refresh")),
+ SLOT_EVENT(EVENT_REFRESH)
+ );
+
+ action_group->add( Gtk::Action::create("properties", Gtk::StockID("gtk-properties")),
+ sigc::mem_fun0(canvas_properties,&studio::CanvasProperties::present)
+ );
+
+ // Preview Quality Menu
+ {
+ int i;
+ action_group->add( Gtk::RadioAction::create(quality_group,"quality-00", _("Use Parametric Renderer")),
+ sigc::bind(
+ sigc::mem_fun(*work_area, &studio::WorkArea::set_quality),
+ 0
+ )
+ );
+ for(i=1;i<=10;i++)
+ {
+ String note;
+ if (i == 1) note = _(" (best)");
+ if (i == 10) note = _(" (fastest)");
+ Glib::RefPtr<Gtk::RadioAction> action(Gtk::RadioAction::create(quality_group,strprintf("quality-%02d",i),
+ strprintf(_("Set Quality to %d"),i) + note));
+ if (i==8) // default quality
+ {
+ action->set_active();
+ work_area->set_quality(i);
+ }
+ action_group->add( action,
+ sigc::bind(
+ sigc::mem_fun(*this, &studio::CanvasView::set_quality),
+ i
+ )
+ );
+ }
+ }
+
+ // Low-Res Quality Menu
+ {
+ int i;
+ for(list<int>::iterator iter = CanvasView::get_pixel_sizes().begin(); iter != CanvasView::get_pixel_sizes().end(); iter++)
+ {
+ i = *iter;
+ Glib::RefPtr<Gtk::RadioAction> action(Gtk::RadioAction::create(low_res_pixel_size_group,strprintf("lowres-pixel-%d",i),
+ strprintf(_("Set Low-Res pixel size to %d"),i)));
+ if(i==2) // default pixel size
+ {
+ action->set_active();
+ work_area->set_low_res_pixel_size(i);
+ }
+ action_group->add( action,
+ sigc::bind(
+ sigc::mem_fun(*work_area, &studio::WorkArea::set_low_res_pixel_size),
+ i
+ )
+ );
+ }
+
+ Glib::RefPtr<Gtk::Action> action;
+
+ action=Gtk::Action::create("decrease-low-res-pixel-size", _("Decrease Low-Res Pixel Size"));
+ action_group->add( action,sigc::mem_fun(this, &studio::CanvasView::decrease_low_res_pixel_size));
+
+ action=Gtk::Action::create("increase-low-res-pixel-size", _("Increase Low-Res Pixel Size"));
+ action_group->add( action, sigc::mem_fun(this, &studio::CanvasView::increase_low_res_pixel_size));
+
+ }
+
+ action_group->add( Gtk::Action::create("play", Gtk::Stock::MEDIA_PLAY),
+ sigc::mem_fun(*this, &studio::CanvasView::play)
+ );
+
+ action_group->add( Gtk::Action::create("dialog-flipbook", _("Preview Window")),
+ sigc::mem_fun0(*preview_dialog, &studio::Dialog_Preview::present)
+ );
+
+ {
+ Glib::RefPtr<Gtk::ToggleAction> action;
+
+ grid_show_toggle = Gtk::ToggleAction::create("toggle-grid-show", _("Show Grid"));
+ grid_show_toggle->set_active(work_area->grid_status());
+ action_group->add(grid_show_toggle, sigc::mem_fun(*this, &studio::CanvasView::toggle_show_grid));
+
+ grid_snap_toggle = Gtk::ToggleAction::create("toggle-grid-snap", _("Snap to Grid"));
+ grid_snap_toggle->set_active(work_area->get_grid_snap());
+ action_group->add(grid_snap_toggle, sigc::mem_fun(*this, &studio::CanvasView::toggle_snap_grid));
+
+ action = Gtk::ToggleAction::create("toggle-guide-show", _("Show Guides"));
+ action->set_active(work_area->get_show_guides());
+ action_group->add(action, sigc::mem_fun(*work_area, &studio::WorkArea::toggle_guide_snap));
+
+ action = Gtk::ToggleAction::create("toggle-low-res", _("Use Low-Res"));
+ action->set_active(work_area->get_low_resolution_flag());
+ action_group->add(action, sigc::mem_fun(*this, &studio::CanvasView::toggle_low_res_pixel_flag));
+
+ action = Gtk::ToggleAction::create("toggle-onion-skin", _("Show Onion Skin"));
+ action->set_active(work_area->get_onion_skin());
+ action_group->add(action, sigc::mem_fun(*this, &studio::CanvasView::toggle_onion_skin));
+ }
+
+ action_group->add( Gtk::Action::create("canvas-zoom-fit", Gtk::StockID("gtk-zoom-fit")),
+ sigc::mem_fun(*work_area, &studio::WorkArea::zoom_fit)
+ );
+ action_group->add( Gtk::Action::create("canvas-zoom-100", Gtk::StockID("gtk-zoom-100")),
+ sigc::mem_fun(*work_area, &studio::WorkArea::zoom_norm)
+ );
+
+ {
+ Glib::RefPtr<Gtk::Action> action;
+
+ action=Gtk::Action::create("seek-next-frame", Gtk::Stock::GO_FORWARD,_("Next Frame"),_("Next Frame"));
+ action_group->add(action,sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_frame),1));
+ action=Gtk::Action::create("seek-prev-frame", Gtk::Stock::GO_BACK,_("Prev Frame"),_("Prev Frame"));
+ action_group->add( action, sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_frame),-1));
+
+ action=Gtk::Action::create("seek-next-second", Gtk::Stock::GO_FORWARD,_("Seek Forward"),_("Seek Forward"));
+ action_group->add(action,sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time),Time(1)));
+ action=Gtk::Action::create("seek-prev-second", Gtk::Stock::GO_BACK,_("Seek Backward"),_("Seek Backward"));
+ action_group->add( action, sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time),Time(-1)));
+
+ action=Gtk::Action::create("seek-end", Gtk::Stock::GOTO_LAST,_("Seek to End"),_("Seek to End"));
+ action_group->add(action,sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time),Time::end()));
+
+ action=Gtk::Action::create("seek-begin", Gtk::Stock::GOTO_FIRST,_("Seek to Begin"),_("Seek to Begin"));
+ action_group->add( action, sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek_time),Time::begin()));
+
+ action=Gtk::Action::create("jump-next-keyframe", Gtk::Stock::GO_FORWARD,_("Jump to Next Keyframe"),_("Jump to Next Keyframe"));
+ action_group->add( action,sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_next_keyframe));
+
+ action=Gtk::Action::create("jump-prev-keyframe", Gtk::Stock::GO_BACK,_("Jump to Prev Keyframe"),_("Jump to Prev Keyframe"));
+ action_group->add( action,sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_prev_keyframe));
+
+ action=Gtk::Action::create("canvas-zoom-in", Gtk::Stock::ZOOM_IN);
+ action_group->add( action,sigc::mem_fun(*work_area, &studio::WorkArea::zoom_in));
+
+ action=Gtk::Action::create("canvas-zoom-out", Gtk::Stock::ZOOM_OUT);
+ action_group->add( action, sigc::mem_fun(*work_area, &studio::WorkArea::zoom_out) );
+
+ action=Gtk::Action::create("time-zoom-in", Gtk::Stock::ZOOM_IN, _("Zoom In on Timeline"));
+ action_group->add( action, sigc::mem_fun(*this, &studio::CanvasView::time_zoom_in) );
+
+ action=Gtk::Action::create("time-zoom-out", Gtk::Stock::ZOOM_OUT, _("Zoom Out on Timeline"));
+ action_group->add( action, sigc::mem_fun(*this, &studio::CanvasView::time_zoom_out) );
+
+ }
+
+ {
+ Glib::RefPtr<Gtk::ToggleAction> action;
+
+#define DUCK_MASK(lower,upper,string) \
+ action=Gtk::ToggleAction::create("mask-" #lower "-ducks", string); \
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_##upper)); \
+ action_group->add(action, \
+ sigc::bind( \
+ sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask), \
+ Duck::TYPE_##upper))
+
+ DUCK_MASK(position,POSITION,_("Show Position Ducks"));
+ DUCK_MASK(tangent,TANGENT,_("Show Tangent Ducks"));
+ DUCK_MASK(vertex,VERTEX,_("Show Vertex Ducks"));
+ DUCK_MASK(radius,RADIUS,_("Show Radius Ducks"));
+ DUCK_MASK(width,WIDTH,_("Show Width Ducks"));
+ DUCK_MASK(angle,ANGLE,_("Show Angle Ducks"));
+
+#undef DUCK_MASK
+ }
+
+ add_accel_group(App::ui_manager()->get_accel_group());
+
+/* // Here is where we add the actions that may have conflicting
+ // keyboard accelerators.
+ {
+ Glib::RefPtr<Gtk::ActionGroup> accel_action_group(Gtk::ActionGroup::create("canvas_view"));
+ Glib::RefPtr<Gtk::Action> action;
+
+ action=Gtk::Action::create("seek-next-frame", Gtk::StockID("gtk-forward"),_("Next Frame"),_("Next Frame"));
+ accel_action_group->add(action,sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek),1));
+
+ action=Gtk::Action::create("seek-prev-frame", Gtk::StockID("gtk-forward"),_("Prev Frame"),_("Prev Frame"));
+ accel_action_group->add( action, sigc::bind(sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::seek),-1));
+
+ action=Gtk::Action::create("jump-next-keyframe", Gtk::StockID("gtk-forward"),_("Jump to Next Keyframe"),_("Jump to Next Keyframe"));
+ accel_action_group->add( action,sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_next_keyframe));
+
+ action=Gtk::Action::create("jump-prev-keyframe", Gtk::StockID("gtk-back"),_("Jump to Prev Keyframe"),_("Jump to Prev Keyframe"));
+ accel_action_group->add( action,sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_prev_keyframe));
+
+ action=Gtk::Action::create("canvas-zoom-in", Gtk::StockID("gtk-zoom-in"));
+ accel_action_group->add( action,sigc::mem_fun(*work_area, &studio::WorkArea::zoom_in));
+
+ action=Gtk::Action::create("canvas-zoom-out", Gtk::StockID("gtk-zoom-out"));
+ accel_action_group->add( action, sigc::mem_fun(*work_area, &studio::WorkArea::zoom_out) );
+
+ action=Gtk::Action::create("time-zoom-in", Gtk::StockID("gtk-zoom-in"), _("Zoom In on Timeline"));
+ accel_action_group->add( action, sigc::mem_fun(*this, &studio::CanvasView::time_zoom_in) );
+
+ action=Gtk::Action::create("time-zoom-out", Gtk::StockID("gtk-zoom-out"), _("Zoom Out on Timeline"));
+ accel_action_group->add( action, sigc::mem_fun(*this, &studio::CanvasView::time_zoom_out) );
+
+ Glib::RefPtr<Gtk::UIManager> accel_ui_manager(Gtk::UIManager::create());
+
+ Glib::ustring ui_info =
+ "
+ <ui>
+ <accelerator action='seek-next-frame' />
+ <accelerator action='seek-prev-frame' />
+ <accelerator action='jump-next-keyframe' />
+ <accelerator action='jump-prev-keyframe' />
+ <accelerator action='canvas-zoom-in' />
+ <accelerator action='canvas-zoom-out' />
+ <accelerator action='time-zoom-in' />
+ <accelerator action='time-zoom-out' />
+ </ui>
+ ";
+
+ accel_ui_manager->add_ui_from_string(ui_info);
+ add_accel_group(accel_ui_manager->get_accel_group());
+
+ accel_ui_manager->insert_action_group(accel_action_group);
+ set_ref_obj("accel_ui_manager",accel_ui_manager);
+ set_ref_obj("accel_action_group",accel_action_group);
+ }
+*/
+
+#if 0
+//
+// //Test some key stuff
+//
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-save"),
+// hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::save))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-save-as"),sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_save_as))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-revert-to-saved"),hide_return(sigc::mem_fun(*get_instance().get(), &studio::Instance::safe_revert))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+//
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("synfig-cvs_add"),sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_add))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("synfig-cvs_update"),sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_update))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("synfig-cvs_commit"),sigc::hide_return(sigc::mem_fun(*get_instance(), &studio::Instance::dialog_cvs_commit))));
+//
+// filemenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Import..."),Gtk::AccelKey('I',Gdk::CONTROL_MASK),sigc::hide_return(sigc::mem_fun(*this, &studio::CanvasView::image_import))));
+// filemenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Render"),Gtk::AccelKey("F9"),
+// sigc::mem_fun(render_settings,&studio::RenderSettings::present)
+// ));
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Preview"),Gtk::AccelKey("F11"),
+// sigc::mem_fun(*this,&CanvasView::on_preview_option)
+// ));
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Sound File"),
+// sigc::mem_fun(*this,&CanvasView::on_audio_option)
+// ));
+//
+// filemenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Options"),Gtk::AccelKey("F12"),
+// sigc::mem_fun(canvas_options,&studio::CanvasOptions::present)
+// ));
+// filemenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Close View"),
+// sigc::hide_return(sigc::mem_fun(*this,&studio::CanvasView::close_view))
+// ));
+// filemenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Close Document"),
+// sigc::hide_return(sigc::mem_fun(*this,&studio::CanvasView::close_document))
+// ));
+//
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-undo"),Gtk::AccelKey('Z',Gdk::CONTROL_MASK),SLOT_EVENT(EVENT_UNDO)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-redo"),Gtk::AccelKey('R',Gdk::CONTROL_MASK),SLOT_EVENT(EVENT_REDO)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-cut"),NOT_IMPLEMENTED_SLOT));
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-copy"),NOT_IMPLEMENTED_SLOT));
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-paste"),NOT_IMPLEMENTED_SLOT));
+// editmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// editmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Select All Ducks"),Gtk::AccelKey('E',Gdk::CONTROL_MASK),sigc::mem_fun(*work_area,&studio::WorkArea::select_all_ducks)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Unselect All Ducks"),Gtk::AccelKey('E',Gdk::CONTROL_MASK),sigc::mem_fun(*work_area,&studio::WorkArea::unselect_all_ducks)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Select All Layers"),Gtk::AccelKey('A',Gdk::CONTROL_MASK),sigc::mem_fun(*this,&CanvasView::on_select_layers)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Unselect All Layers"),Gtk::AccelKey('D',Gdk::CONTROL_MASK),sigc::mem_fun(*this,&CanvasView::on_unselect_layers)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+//
+// //editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-stop"),Gtk::AccelKey(GDK_Escape,static_cast<Gdk::ModifierType>(0)),sigc::hide_return(sigc::mem_fun(*this, &studio::CanvasView::stop))));
+// //editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-refresh"),Gtk::AccelKey('k',Gdk::CONTROL_MASK),sigc::hide_return(sigc::mem_fun(*this, &studio::CanvasView::on_refresh_pressed))));
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-stop"),Gtk::AccelKey(GDK_Escape,static_cast<Gdk::ModifierType>(0)),SLOT_EVENT(EVENT_STOP)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-refresh"),Gtk::AccelKey('k',Gdk::CONTROL_MASK),SLOT_EVENT(EVENT_REFRESH)));
+// editmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// editmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-properties"),Gtk::AccelKey("F8"),
+// sigc::mem_fun(canvas_properties,&studio::CanvasProperties::present)
+// ));
+//
+// build_new_layer_menu(newlayermenu);
+// layermenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("New"),newlayermenu));
+//
+//
+// {
+// synfigapp::Action::ParamList param_list;
+// param_list.add("canvas",Canvas::Handle(get_canvas()));
+// param_list.add("canvas_interface",canvas_interface());
+// add_actions_to_menu(&canvasmenu, param_list,synfigapp::Action::CATEGORY_CANVAS);
+// }
+//
+//
+// //canvasmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Keyframe Dialog"),sigc::mem_fun(keyframe_dialog,&studio::Dialog_Keyframe::present)));
+//
+// // Duck Mask Menu
+// if(1)
+// {
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::TearoffMenuElem());
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Position Ducks"),Gtk::AccelKey('1',Gdk::MOD1_MASK)));
+// duck_mask_position=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_position->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_POSITION));
+// duck_mask_position->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_POSITION
+// )
+// );
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Vertex Ducks"),Gtk::AccelKey('2',Gdk::MOD1_MASK)));
+// duck_mask_vertex=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_vertex->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_VERTEX));
+// duck_mask_vertex->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_VERTEX
+// )
+// );
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Tangent Ducks"),Gtk::AccelKey('3',Gdk::MOD1_MASK)));
+// duck_mask_tangent=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_tangent->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_TANGENT));
+// duck_mask_tangent->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_TANGENT
+// )
+// );
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Radius Ducks"),Gtk::AccelKey('4',Gdk::MOD1_MASK)));
+// duck_mask_radius=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_radius->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_RADIUS));
+// duck_mask_radius->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_RADIUS
+// )
+// );
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Width Ducks"),Gtk::AccelKey('5',Gdk::MOD1_MASK)));
+// duck_mask_width=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_width->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_WIDTH));
+// duck_mask_width->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_WIDTH
+// )
+// );
+//
+// duckmaskmenu.items().push_back(Gtk::Menu_Helpers::CheckMenuElem(_("Angle Ducks"),Gtk::AccelKey('6',Gdk::MOD1_MASK)));
+// duck_mask_angle=static_cast<Gtk::CheckMenuItem*>(&duckmaskmenu.items().back());
+// duck_mask_angle->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_ANGLE));
+// duck_mask_angle->signal_toggled().connect(
+// sigc::bind(
+// sigc::mem_fun(*this, &studio::CanvasView::toggle_duck_mask),
+// Duck::TYPE_ANGLE
+// )
+// );
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Mask Ducks"),duckmaskmenu));
+// }
+//
+// // Preview Quality Menu
+// if(1)
+// {
+// qualitymenu.items().push_back(Gtk::Menu_Helpers::TearoffMenuElem());
+// int i;
+// qualitymenu.items().push_back(Gtk::Menu_Helpers::MenuElem(strprintf(_("Use Parametric Renderer"),0),
+// sigc::bind(
+// sigc::mem_fun(*work_area, &studio::WorkArea::set_quality),
+// 0
+// )
+// ));
+// for(i=1;i<=10;i++)
+// {
+// qualitymenu.items().push_back(Gtk::Menu_Helpers::MenuElem(strprintf(_("Set Quality to %d"),i),Gtk::AccelKey('0'+(i%10),Gdk::CONTROL_MASK),
+// sigc::bind(
+// sigc::mem_fun(*work_area, &studio::WorkArea::set_quality),
+// i
+// )
+// ));
+// }
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Preview Quality"),qualitymenu));
+// }
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::Stock::MEDIA_PLAY,
+// sigc::mem_fun(*this, &studio::CanvasView::play)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("Flipbook Dialog"),
+// sigc::mem_fun(*preview_dialog, &studio::Dialog_Preview::present)));
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Toggle Grid Show"),Gtk::AccelKey('g',Gdk::CONTROL_MASK),
+// sigc::mem_fun(*work_area, &studio::WorkArea::toggle_grid)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Toggle Grid Snap"),Gtk::AccelKey('l',Gdk::CONTROL_MASK),
+// sigc::mem_fun(*work_area, &studio::WorkArea::toggle_grid_snap)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Toggle Guide Snap"),Gtk::AccelKey('k',Gdk::CONTROL_MASK),
+// sigc::mem_fun(*work_area, &studio::WorkArea::toggle_guide_snap)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Toggle Low-Res"),Gtk::AccelKey('`',Gdk::CONTROL_MASK),
+// sigc::mem_fun(*work_area, &studio::WorkArea::toggle_low_resolution_flag)));
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-in"),Gtk::AccelKey('=',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*work_area, &studio::WorkArea::zoom_in)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-out"),Gtk::AccelKey('-',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*work_area, &studio::WorkArea::zoom_out)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-fit"),
+// sigc::mem_fun(*work_area, &studio::WorkArea::zoom_fit)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-100"),Gtk::AccelKey('`',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*work_area, &studio::WorkArea::zoom_norm)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-in"),Gtk::AccelKey('+',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*this, &studio::CanvasView::time_zoom_in)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-zoom-out"),Gtk::AccelKey('_',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*this, &studio::CanvasView::time_zoom_out)));
+//
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Jump to Next Keyframe"),Gtk::AccelKey(']',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_next_keyframe)));
+// viewmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Jump to Prev Keyframe"),Gtk::AccelKey('[',static_cast<Gdk::ModifierType>(0)),
+// sigc::mem_fun(*canvas_interface().get(), &synfigapp::CanvasInterface::jump_to_prev_keyframe)));
+//
+// mainmenu.items().push_back(Gtk::Menu_Helpers::TearoffMenuElem());
+// mainmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_File"),filemenu));
+// mainmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Edit"),editmenu));
+// mainmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_View"),viewmenu));
+// mainmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Canvas"),canvasmenu));
+// mainmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Layer"),layermenu));
+//
+// mainmenu.accelerate(*this);
+//
+///*
+// {
+//
+// trackmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("New Waypoint"),NOT_IMPLEMENTED_SLOT));
+// trackmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Delete Waypoint"),NOT_IMPLEMENTED_SLOT));
+// trackmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Export"),NOT_IMPLEMENTED_SLOT));
+// trackmenu.items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+// trackmenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Properties"),NOT_IMPLEMENTED_SLOT));
+// }
+//*/
+// mainmenu.show();
+// filemenu.show();
+// editmenu.show();
+// canvasmenu.show();
+// layermenu.show();
+//
+// keyframemenu.items().push_back(Gtk::Menu_Helpers::StockMenuElem(Gtk::StockID("gtk-properties"),
+// sigc::mem_fun(*this,&studio::CanvasView::show_keyframe_dialog)
+// ));
+//
+//
+// get_accel_group()->unlock();
+//
+// //Set the accelerator paths for all the menus
+// filemenu.set_accel_path("<synfig>/File");
+// editmenu.set_accel_path("<synfig>/Edit");
+// layermenu.set_accel_path("<synfig>/Layer");
+// //mainmenu.set_accel_path("<synfig-main>");
+// canvasmenu.set_accel_path("<synfig>/Canvas");
+// viewmenu.set_accel_path("<synfig>/View");
+// duckmaskmenu.set_accel_path("<synfig>/DuckMask");
+#endif
+}
+
+void
+CanvasView::on_select_layers()
+{
+ Canvas::Handle canvas(get_canvas());
+ for (CanvasBase::iterator iter = canvas->begin(); iter != canvas->end(); iter++)
+ layer_tree->select_all_children_layers(*iter);
+}
+
+void
+CanvasView::on_unselect_layers()
+{
+ layer_tree->clear_selected_layers();
+}
+
+void
+CanvasView::show_keyframe_dialog()
+{
+ Glib::RefPtr<Gtk::TreeSelection> selection(keyframe_tree->get_selection());
+ if(selection->get_selected())
+ {
+ Gtk::TreeRow row(*selection->get_selected());
+
+ Keyframe keyframe(row[keyframe_tree->model.keyframe]);
+
+ keyframe_dialog.set_keyframe(keyframe);
+ keyframe_dialog.present();
+ }
+}
+
+void
+CanvasView::add_layer(synfig::String x)
+{
+ Canvas::Handle canvas;
+
+ synfigapp::SelectionManager::LayerList layer_list(get_selection_manager()->get_selected_layers());
+
+ int target_depth(0);
+
+ if(layer_list.empty())
+ {
+ canvas=get_canvas();
+ }
+ else
+ {
+ canvas=(*layer_list.begin())->get_canvas();
+ target_depth=canvas->get_depth(*layer_list.begin());
+ }
+
+ Layer::Handle layer(canvas_interface()->add_layer_to(x,canvas,target_depth));
+ if(layer)
+ {
+ get_selection_manager()->clear_selected_layers();
+ get_selection_manager()->set_selected_layer(layer);
+ }
+}
+
+void
+CanvasView::popup_layer_menu(synfig::Layer::Handle layer)
+{
+ //Gtk::Menu* menu(manage(new Gtk::Menu));
+ Gtk::Menu* menu(¶mmenu);
+ menu->items().clear();
+
+ synfigapp::Action::ParamList param_list;
+ param_list.add("time",canvas_interface()->get_time());
+ param_list.add("canvas",Canvas::Handle(layer->get_canvas()));
+ param_list.add("canvas_interface",canvas_interface());
+ param_list.add("layer",layer);
+
+ //Gtk::Menu *newlayers(manage(new Gtk::Menu()));
+ //build_new_layer_menu(*newlayers);
+
+ //parammenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("New Layer"),*newlayers));
+
+ if(layer->get_name()=="PasteCanvas")
+ {
+ menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("Select All Children"),
+ sigc::bind(
+ sigc::mem_fun(
+ *layer_tree,
+ &studio::LayerTree::select_all_children_layers
+ ),
+ layer
+ )
+ ));
+ }
+
+ add_actions_to_menu(menu, param_list,synfigapp::Action::CATEGORY_LAYER);
+
+ menu->popup(3,gtk_get_current_event_time());
+}
+
+void
+CanvasView::register_layer_type(synfig::Layer::Book::value_type &/*lyr*/,std::map<synfig::String,Gtk::Menu*>* /*category_map*/)
+{
+/* if(lyr.second.category==CATEGORY_DO_NOT_USE)
+ return;
+
+ if(category_map->count(lyr.second.category)==0)
+ (*category_map)[lyr.second.category]=manage(new Gtk::Menu());
+
+ (*category_map)[lyr.second.category]->items().push_back(Gtk::Menu_Helpers::MenuElem(lyr.second.local_name,
+ sigc::hide_return(
+ sigc::bind(
+ sigc::mem_fun(*this,&studio::CanvasView::add_layer),
+ lyr.first
+ )
+ )
+ ));
+*/
+}
+
+void
+CanvasView::build_new_layer_menu(Gtk::Menu &/*menu*/)
+{
+/*
+ std::map<synfig::String,Gtk::Menu*> category_map;
+
+ std::for_each(
+ synfig::Layer::book().begin(),
+ synfig::Layer::book().end(),
+ sigc::bind(
+ sigc::mem_fun(
+ *this,
+ &studio::CanvasView::register_layer_type
+ ),
+ &category_map
+ )
+ );
+
+ menu.items().clear();
+ menu.items().push_back(Gtk::Menu_Helpers::TearoffMenuElem());
+
+ std::map<synfig::String,Gtk::Menu*>::iterator iter;
+ for(iter=category_map.begin();iter!=category_map.end();++iter)
+ menu.items().push_back(Gtk::Menu_Helpers::MenuElem(iter->first,*iter->second));
+
+ menu.show();
+*/
+}
+
+void
+CanvasView::popup_main_menu()
+{
+ //mainmenu.popup(0,gtk_get_current_event_time());
+ Gtk::Menu* menu = dynamic_cast<Gtk::Menu*>(App::ui_manager()->get_widget("/menu-main"));
+ if(menu)
+ {
+ //menu->set_accel_group(App::ui_manager()->get_accel_group());
+ //menu->accelerate(*this);
+ menu->popup(0,gtk_get_current_event_time());
+ }
+}
+
+void
+CanvasView::on_refresh_pressed()
+{
+ rebuild_tables();
+ rebuild_ducks();
+ work_area->queue_render_preview();
+}
+
+void
+CanvasView::workarea_layer_selected(synfig::Layer::Handle layer)
+{
+ get_selection_manager()->clear_selected_layers();
+ if(layer)
+ get_selection_manager()->set_selected_layer(layer);
+}
+
+void
+CanvasView::refresh_rend_desc()
+{
+ current_time_widget->set_fps(get_canvas()->rend_desc().get_frame_rate());
+
+ //????
+ //synfig::info("Canvasview: Refreshing render desc info");
+ if(!get_time().is_equal(time_adjustment().get_value()))
+ {
+ time_adjustment().set_value(get_time());
+ time_adjustment().value_changed();
+ }
+
+ Time length(get_canvas()->rend_desc().get_time_end()-get_canvas()->rend_desc().get_time_start());
+ if(length<DEFAULT_TIME_WINDOW_SIZE)
+ {
+ time_window_adjustment().set_page_increment(length);
+ time_window_adjustment().set_page_size(length);
+ }
+ else
+ {
+ time_window_adjustment().set_page_increment(DEFAULT_TIME_WINDOW_SIZE);
+ time_window_adjustment().set_page_size(DEFAULT_TIME_WINDOW_SIZE);
+ }
+
+ //set the FPS of the timeslider
+ timeslider->set_global_fps(get_canvas()->rend_desc().get_frame_rate());
+
+ //set the beginning and ending time of the time slider
+ Time begin_time=get_canvas()->rend_desc().get_time_start();
+ Time end_time=get_canvas()->rend_desc().get_time_end();
+
+ // Setup the time_window adjustment
+ time_window_adjustment().set_lower(begin_time);
+ time_window_adjustment().set_upper(end_time);
+ time_window_adjustment().set_step_increment(synfig::Time(1.0/get_canvas()->rend_desc().get_frame_rate()));
+
+ //Time length(get_canvas()->rend_desc().get_time_end()-get_canvas()->rend_desc().get_time_start());
+ if(length < time_window_adjustment().get_page_size())
+ {
+ time_window_adjustment().set_page_increment(length);
+ time_window_adjustment().set_page_size(length);
+ }
+
+ /*synfig::info("w: %p - [%.3f,%.3f] (%.3f,%.3f) child: %p\n",
+ &time_window_adjustment_, time_window_adjustment_.get_lower(),
+ time_window_adjustment_.get_upper(),time_window_adjustment_.get_value(),
+ time_window_adjustment_.get_page_size(),time_window_adjustment_.get_child_adjustment()
+ );*/
+
+ time_window_adjustment().changed(); //only non-value stuff was changed
+
+ // Setup the time adjustment
+
+ //NOTE THESE TWO SHOULD BE CHANGED BY THE changed() CALL ABOVE
+ //time_adjustment().set_lower(time_window_adjustment().get_value());
+ //time_adjustment().set_upper(time_window_adjustment().get_value()+time_window_adjustment().get_page_size());
+
+// time_adjustment().set_lower(get_canvas()->rend_desc().get_time_start());
+// time_adjustment().set_upper(get_canvas()->rend_desc().get_time_end());
+ time_adjustment().set_step_increment(synfig::Time(1.0/get_canvas()->rend_desc().get_frame_rate()));
+ time_adjustment().set_page_increment(synfig::Time(1.0));
+ time_adjustment().set_page_size(0);
+
+ time_adjustment().changed();
+
+ /*synfig::info("w: %p - [%.3f,%.3f] (%.3f,%.3f) child: %p\n",
+ &time_window_adjustment_, time_window_adjustment_.get_lower(),
+ time_window_adjustment_.get_upper(),time_window_adjustment_.get_value(),
+ time_window_adjustment_.get_page_size(),time_window_adjustment_.get_child_adjustment()
+ ); */
+
+ if(begin_time==end_time)
+ {
+ hide_timebar();
+ }
+ else
+ {
+ show_timebar();
+ }
+
+ //clamp time to big bounds...
+ if(time_window_adjustment().get_value() < begin_time)
+ {
+ time_window_adjustment().set_value(begin_time);
+ time_window_adjustment().value_changed();
+ }
+
+ if(time_window_adjustment().get_value() + time_window_adjustment().get_page_size() > end_time)
+ {
+ time_window_adjustment().set_value(end_time - time_window_adjustment().get_page_size());
+ time_window_adjustment().value_changed();
+ }
+
+ if(time_adjustment().get_value() < begin_time)
+ {
+ time_adjustment().set_value(begin_time);
+ time_adjustment().value_changed();
+ }
+
+ if(time_adjustment().get_value() > end_time)
+ {
+ time_adjustment().set_value(end_time);
+ time_adjustment().value_changed();
+ }
+
+ /*synfig::info("Time stats: \n"
+ "w: %p - [%.3f,%.3f] (%.3f,%.3f) child: %p\n"
+ "t: %p - [%.3f,%.3f] %.3f",
+ &time_window_adjustment_, time_window_adjustment_.get_lower(),
+ time_window_adjustment_.get_upper(),time_window_adjustment_.get_value(),
+ time_window_adjustment_.get_page_size(),time_window_adjustment_.get_child_adjustment(),
+ &time_adjustment_,time_adjustment_.get_lower(),time_adjustment_.get_upper(),
+ time_adjustment_.get_value()
+ );*/
+
+ work_area->queue_render_preview();
+}
+
+bool
+CanvasView::close_view()
+{
+ if(get_instance()->get_visible_canvases()==1)
+ close_instance();
+ else
+ hide();
+ return false;
+}
+
+static bool _close_instance(etl::handle<Instance> instance)
+{
+ etl::handle<Instance> argh(instance);
+ instance->safe_close();
+ synfig::info("closed");
+ return false;
+}
+
+bool
+CanvasView::close_instance()
+{
+#ifdef SINGLE_THREADED
+ if (get_work_area()->get_updating())
+ {
+ get_work_area()->stop_updating(true); // stop and mark as cancelled
+
+ // give the workarea chances to stop updating
+ Glib::signal_timeout().connect(
+ sigc::mem_fun(*this, &CanvasView::close_instance),
+ 250);
+ }
+ else
+#endif
+ Glib::signal_timeout().connect(
+ sigc::bind(sigc::ptr_fun(_close_instance),
+ (etl::handle<Instance>)get_instance()),
+ 250);
+ return false;
+}
+
+handle<CanvasView>
+CanvasView::create(etl::loose_handle<Instance> instance, etl::handle<synfig::Canvas> canvas)
+{
+ etl::handle<studio::CanvasView> view(new CanvasView(instance,instance->synfigapp::Instance::find_canvas_interface(canvas)));
+ instance->canvas_view_list().push_front(view);
+ instance->signal_canvas_view_created()(view.get());
+ return view;
+}
+
+void
+CanvasView::update_title()
+{
+ string title;
+
+ title = strprintf("%s%s\"%s\"",
+ (
+ get_instance()->get_action_count()
+ ? "*"
+ : ""
+ ), (
+ get_instance()->has_real_filename()
+ ? (etl::basename(get_instance()->get_file_name()) + " : ").c_str()
+ : ""
+ ), (
+ get_canvas()->get_name().empty()
+ ? get_canvas()->get_id().c_str()
+ : get_canvas()->get_name().c_str()
+ ));
+
+ if(get_instance()->synfigapp::Instance::in_repository())
+ {
+ title+=" (CVS";
+ if(get_instance()->synfigapp::Instance::is_modified())
+ title+=_("-MODIFIED");
+ if(get_instance()->synfigapp::Instance::is_updated())
+ title+=_("-UPDATED");
+ title+=')';
+ }
+
+ if(get_canvas()->is_root())
+ title+=_(" (Root)");
+
+ set_title(title);
+}
+
+void
+CanvasView::on_hide()
+{
+ smach_.egress();
+ Gtk::Window::on_hide();
+}
+
+void
+CanvasView::present()
+{
+ grab_focus();//on_focus_in_event(0);
+ Gtk::Window::present();
+}
+
+bool
+CanvasView::on_focus_in_event(GdkEventFocus*x)
+{
+ if(studio::App::get_selected_canvas_view()!=this)
+ {
+ if(studio::App::get_selected_canvas_view())
+ {
+ studio::App::get_selected_canvas_view()->get_smach().process_event(EVENT_YIELD_TOOL_OPTIONS);
+ App::ui_manager()->remove_action_group(App::get_selected_canvas_view()->action_group);
+ }
+
+ get_smach().process_event(EVENT_REFRESH_TOOL_OPTIONS);
+
+ studio::App::set_selected_canvas_view(this);
+
+ App::ui_manager()->insert_action_group(action_group);
+ }
+
+ // HACK ... Questionable...?
+ if(x)
+ return Gtk::Window::on_focus_in_event(x);
+
+ return true;
+}
+
+bool
+CanvasView::on_focus_out_event(GdkEventFocus*x)
+{
+ //App::ui_manager()->remove_action_group(action_group);
+ //App::ui_manager()->ensure_update();
+ return Gtk::Window::on_focus_out_event(x);
+}
+
+void
+CanvasView::refresh_tables()
+{
+// if(layer_tree_store_)layer_tree_store_->refresh();
+// if(children_tree_store_)children_tree_store_->refresh();
+}
+
+void
+CanvasView::rebuild_tables()
+{
+// layer_tree_store_->rebuild();
+// children_tree_store_->rebuild();
+}
+
+void
+CanvasView::build_tables()
+{
+// layer_tree_store_->rebuild();
+// children_tree_store_->rebuild();
+}
+
+void
+CanvasView::on_layer_toggle(synfig::Layer::Handle layer)
+{
+ synfigapp::Action::Handle action(synfigapp::Action::create("LayerActivate"));
+ assert(action);
+
+ if(!action)
+ return;
+
+ action->set_param("canvas",Canvas::Handle(layer->get_canvas()));
+ if(!action->set_param("canvas_interface",canvas_interface()))
+// if(!action->set_param("canvas_interface",get_instance()->find_canvas_interface(layer->get_canvas())))
+ synfig::error("LayerActivate didn't like CanvasInterface...?");
+ action->set_param("time",get_time());
+ action->set_param("layer",layer);
+ action->set_param("new_status",!layer->active());
+
+ assert(action->is_ready());
+
+ canvas_interface()->get_instance()->perform_action(action);
+}
+
+void
+CanvasView::popup_param_menu(synfigapp::ValueDesc value_desc, float location, bool bezier)
+{
+ parammenu.items().clear();
+ get_instance()->make_param_menu(¶mmenu,get_canvas(),value_desc,location,bezier);
+
+ parammenu.popup(3,gtk_get_current_event_time());
+}
+
+void
+CanvasView::add_actions_to_menu(Gtk::Menu *menu, const synfigapp::Action::ParamList ¶m_list,synfigapp::Action::Category category)const
+{
+ get_instance()->add_actions_to_menu(menu, param_list, category);
+}
+
+bool
+CanvasView::on_layer_user_click(int button, Gtk::TreeRow /*row*/, LayerTree::ColumnID /*column_id*/)
+{
+ switch(button)
+ {
+ case 3:
+ {
+ Gtk::MenuItem* menu = dynamic_cast<Gtk::MenuItem*>(App::ui_manager()->get_widget("/menu-main/menu-layer"));
+ if(menu && menu->get_submenu())
+ {
+ //menu->set_accel_group(App::ui_manager()->get_accel_group());
+ //menu->accelerate(*this);
+ menu->get_submenu()->popup(button,gtk_get_current_event_time());
+ }
+
+ #if 0
+ bool multiple_selected=true;
+
+ if(layer_tree->get_selection()->count_selected_rows()<=1)
+ multiple_selected=false;
+
+ // If the clicked row is not selected, then unselect
+ // everything that isn't selected and select this row
+ if(multiple_selected && !layer_tree->get_selection()->is_selected(row))
+ {
+ layer_tree->get_selection()->unselect_all();
+ layer_tree->get_selection()->select(row);
+ multiple_selected=false;
+ }
+
+ if(column_id==COLUMNID_TIME_TRACK)
+ return false;
+
+ //synfigapp::ValueDesc value_desc(row[layer_param_tree_model.value_desc]);
+ //ValueNode::Handle value_node(row[layer_param_tree_model.value_node]);
+ //ValueNode::Handle parent_value_node;
+ //ValueBase value=row[layer_param_tree_model.value];
+
+ //if(row.parent())
+ //{
+ // parent_value_node=(*row.parent())[layer_tree_model.value_node];
+ //}
+
+ {
+ Layer::Handle layer(row[layer_tree_model.layer]);
+ synfigapp::Action::ParamList param_list;
+ param_list.add("time",canvas_interface()->get_time());
+ param_list.add("canvas",Canvas::Handle(row[layer_tree_model.canvas]));
+ param_list.add("canvas_interface",canvas_interface());
+ if(!multiple_selected)
+ param_list.add("layer",layer);
+ else
+ {
+ synfigapp::SelectionManager::LayerList layer_list(get_selection_manager()->get_selected_layers());
+ synfigapp::SelectionManager::LayerList::iterator iter;
+
+ for(iter=layer_list.begin();iter!=layer_list.end();++iter)
+ param_list.add("layer",Layer::Handle(*iter));
+ }
+
+ parammenu.items().clear();
+
+ Gtk::Menu *newlayers(manage(new Gtk::Menu()));
+ // do we need this? the code is all #ifdef'ed out anyway
+ // newlayers->signal_hide().connect(sigc::bind(sigc::ptr_fun(&delete_widget), newlayers));
+ build_new_layer_menu(*newlayers);
+
+ parammenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("New Layer"),*newlayers));
+ if(!multiple_selected && layer->get_name()=="PasteCanvas")
+ {
+ parammenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Select All Children"),
+ sigc::bind(
+ sigc::mem_fun(
+ *layer_tree,
+ &studio::LayerTree::select_all_children_layers
+ ),
+ layer
+ )
+ ));
+ }
+
+ add_actions_to_menu(¶mmenu, param_list,synfigapp::Action::CATEGORY_LAYER);
+ parammenu.popup(button,gtk_get_current_event_time());
+ return true;
+ }
+/*
+ else if(column_id==LayerTree::COLUMNID_TIME_TRACK && value_node && handle<synfig::ValueNode_Animated>::cast_dynamic(value_node))
+ {
+ // Right-click on time track with animated
+// trackmenu.popup(0,0);
+ return true;
+ }
+ else
+ {
+ if(!multiple_selected)
+ {
+ popup_param_menu(value_desc);
+ return true;
+ }
+ else
+ {
+#warning update me!
+#if 0
+ parammenu.items().clear();
+ parammenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Connect"),
+ hide_return(sigc::mem_fun(*canvas_interface().get(),&synfigapp::CanvasInterface::connect_selected_layer_params))
+ ));
+ parammenu.items().push_back(Gtk::Menu_Helpers::MenuElem(_("Disconnect"),
+ hide_return(sigc::mem_fun(*canvas_interface().get(),&synfigapp::CanvasInterface::disconnect_selected_layer_params))
+ ));
+ parammenu.popup(0,0);
+#endif
+ }
+ return true;
+ }
+ */
+#endif
+}
+ return true;
+ break;
+
+ default:
+ return false;
+ break;
+ }
+}
+
+bool
+CanvasView::on_children_user_click(int button, Gtk::TreeRow row, ChildrenTree::ColumnID column_id)
+{
+ switch(button)
+ {
+ case 3:
+ {
+ if(column_id==COLUMNID_TIME_TRACK)
+ return false;
+ if(!(bool)row[children_tree_model.is_canvas])
+ {
+ synfigapp::ValueDesc value_desc=row[children_tree_model.value_desc];
+ if (!value_desc)
+ {
+ //! \todo fix properly -- what is the child dialog for?
+ synfig::info("preventing child dialog right-click crash");
+ return true;
+ }
+ assert(value_desc);
+ popup_param_menu(value_desc);
+ return true;
+ }
+ }
+ return true;
+ break;
+
+ default:
+ return false;
+ break;
+ }
+}
+
+bool
+CanvasView::on_keyframe_tree_event(GdkEvent *event)
+{
+ switch(event->type)
+ {
+ case GDK_BUTTON_PRESS:
+ switch(event->button.button)
+ {
+ case 3:
+ {
+ //keyframemenu.popup(event->button.button,gtk_get_current_event_time());
+ return true;
+ }
+ break;
+ }
+ break;
+ case GDK_MOTION_NOTIFY:
+ break;
+ case GDK_BUTTON_RELEASE:
+ break;
+ default:
+ break;
+ }
+ return false;
+}
+
+void
+CanvasView::refresh_time_window()
+{
+ //THESE SHOULD AUTOMATICALLY BE TAKEN CARE OF
+ //time_adjustment().set_lower(time_window_adjustment().get_value());
+ //time_adjustment().set_upper(time_window_adjustment().get_value()+time_window_adjustment().get_page_size());
+
+ time_adjustment().set_page_increment(1.0); // One second
+ time_adjustment().set_page_size(0);
+
+ if(get_canvas())
+ time_adjustment().set_step_increment(1.0/get_canvas()->rend_desc().get_frame_rate());
+ time_adjustment().changed();
+
+ //NOTE THIS SHOULD HOOK INTO THE CORRECT SIGNALS...
+ if(children_tree)
+ children_tree->queue_draw();
+}
+
+void
+CanvasView::on_time_changed()
+{
+ Time time(get_time());
+
+ current_time_widget->set_value(time);
+ try {
+ get_canvas()->keyframe_list().find(time);
+ current_time_widget->modify_text(Gtk::STATE_NORMAL,Gdk::Color("#FF0000"));
+ }catch(...){
+ current_time_widget->modify_text(Gtk::STATE_NORMAL,Gdk::Color("#000000"));
+ }
+
+ if(get_time() != time_adjustment().get_value())
+ {
+ //Recenters the window, causing it to jump (possibly undesirably... but whatever)
+ if(time < time_window_adjustment().get_value() ||
+ time > time_window_adjustment().get_value()+time_window_adjustment().get_page_size())
+ {
+ time_window_adjustment().set_value(
+ time-time_window_adjustment().get_page_size()/2
+ );
+ }
+ time_adjustment().set_value(time);
+ time_adjustment().value_changed();
+
+ // Shouldn't these trees just hook into
+ // the time changed signal...?
+ //YES THEY SHOULD...
+ if(layer_tree)layer_tree->queue_draw();
+ if(children_tree)children_tree->queue_draw();
+ }
+}
+
+void
+CanvasView::time_zoom_in()
+{
+ float frame_rate = get_canvas()->rend_desc().get_frame_rate();
+ Time min_page_size = 2/frame_rate;
+
+ time_window_adjustment().set_page_size(time_window_adjustment().get_page_size()*0.75);
+ if (time_window_adjustment().get_page_size() < min_page_size)
+ time_window_adjustment().set_page_size(min_page_size);
+ time_window_adjustment().set_page_increment(time_window_adjustment().get_page_size());
+ time_window_adjustment().changed();
+
+ refresh_time_window();
+}
+
+void
+CanvasView::time_zoom_out()
+{
+ Time length = (get_canvas()->rend_desc().get_time_end() -
+ get_canvas()->rend_desc().get_time_start());
+
+ time_window_adjustment().set_page_size(time_window_adjustment().get_page_size()/0.75);
+ if (time_window_adjustment().get_page_size() > length)
+ time_window_adjustment().set_page_size(length);
+ time_window_adjustment().set_page_increment(time_window_adjustment().get_page_size());
+ time_window_adjustment().changed();
+
+ refresh_time_window();
+}
+
+void
+CanvasView::time_was_changed()
+{
+ synfig::Time time((synfig::Time)(double)time_adjustment().get_value());
+ set_time(time);
+}
+
+void
+CanvasView::on_edited_value(synfigapp::ValueDesc value_desc,synfig::ValueBase new_value)
+{
+ canvas_interface()->change_value(value_desc,new_value);
+}
+
+/*
+void
+CanvasView::on_children_edited_value(const Glib::ustring&path_string,synfig::ValueBase value)
+{
+ Gtk::TreePath path(path_string);
+
+ const Gtk::TreeRow row = *(children_tree->get_model()->get_iter(path));
+
+ assert((bool)row[children_tree_model.is_value_node]);
+
+ synfigapp::ValueDesc value_desc=row[children_tree_model.value_desc];
+ assert(value_desc);
+
+ on_edited_value(value_desc,value);
+}
+*/
+
+void
+CanvasView::on_id_changed()
+{
+ update_title();
+}
+
+void
+CanvasView::on_mode_changed(synfigapp::CanvasInterface::Mode mode)
+{
+ // If the animate flag was set in mode...
+ Gtk::IconSize iconsize=Gtk::IconSize::from_name("synfig-small_icon");
+ if(mode&synfigapp::MODE_ANIMATE)
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("gtk-no"),iconsize));
+ animatebutton->remove();
+ animatebutton->add(*icon);
+ tooltips.set_tip(*animatebutton,_("In Animate Editing Mode"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+ else
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("gtk-yes"),iconsize));
+ animatebutton->remove();
+ animatebutton->add(*icon);
+ tooltips.set_tip(*animatebutton,_("Not in Animate Editing Mode"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+
+ if((mode&synfigapp::MODE_ANIMATE_FUTURE) && (mode&synfigapp::MODE_ANIMATE_PAST))
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("synfig-keyframe_lock_all"),Gtk::ICON_SIZE_BUTTON));
+ keyframebutton->remove();
+ keyframebutton->add(*icon);
+ tooltips.set_tip(*keyframebutton,_("All Keyframes Locked"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+ else if((mode&synfigapp::MODE_ANIMATE_FUTURE) && !(mode&synfigapp::MODE_ANIMATE_PAST))
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("synfig-keyframe_lock_future"),Gtk::ICON_SIZE_BUTTON));
+ keyframebutton->remove();
+ keyframebutton->add(*icon);
+ tooltips.set_tip(*keyframebutton,_("Future Keyframes Locked"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+ else if(!(mode&synfigapp::MODE_ANIMATE_FUTURE) && (mode&synfigapp::MODE_ANIMATE_PAST))
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("synfig-keyframe_lock_past"),Gtk::ICON_SIZE_BUTTON));
+ keyframebutton->remove();
+ keyframebutton->add(*icon);
+ tooltips.set_tip(*keyframebutton,_("Past Keyframes Locked"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+ else if(!(mode&synfigapp::MODE_ANIMATE_FUTURE) && !(mode&synfigapp::MODE_ANIMATE_PAST))
+ {
+ Gtk::Image *icon;
+ icon=manage(new Gtk::Image(Gtk::StockID("synfig-keyframe_lock_none"),Gtk::ICON_SIZE_BUTTON));
+ keyframebutton->remove();
+ keyframebutton->add(*icon);
+ tooltips.set_tip(*keyframebutton,_("No Keyframes Locked"));
+ icon->set_padding(0,0);
+ icon->show();
+ }
+
+ work_area->queue_draw();
+}
+
+void
+CanvasView::on_animate_button_pressed()
+{
+ if(get_mode()&synfigapp::MODE_ANIMATE)
+ set_mode(get_mode()-synfigapp::MODE_ANIMATE);
+ else
+ set_mode(get_mode()|synfigapp::MODE_ANIMATE);
+}
+
+void
+CanvasView::on_keyframe_button_pressed()
+{
+ synfigapp::CanvasInterface::Mode mode(get_mode());
+
+ // future && past --> past
+ if((mode&synfigapp::MODE_ANIMATE_FUTURE) && (mode&synfigapp::MODE_ANIMATE_PAST))
+ set_mode(get_mode()-synfigapp::MODE_ANIMATE_FUTURE);
+ // past --> future
+ else if(!(mode&synfigapp::MODE_ANIMATE_FUTURE) && (mode&synfigapp::MODE_ANIMATE_PAST))
+ set_mode((get_mode()-synfigapp::MODE_ANIMATE_PAST)|synfigapp::MODE_ANIMATE_FUTURE);
+ // future --> (nothing)
+ else if((mode&synfigapp::MODE_ANIMATE_FUTURE) && !(mode&synfigapp::MODE_ANIMATE_PAST))
+ set_mode(get_mode()-synfigapp::MODE_ANIMATE_FUTURE);
+ // (nothing) --> future && past
+ else if(!(mode&synfigapp::MODE_ANIMATE_FUTURE) && !(mode&synfigapp::MODE_ANIMATE_PAST))
+ set_mode(get_mode()|synfigapp::MODE_ANIMATE_FUTURE|synfigapp::MODE_ANIMATE_PAST);
+}
+
+bool
+CanvasView::duck_change_param(const synfig::Point &value,synfig::Layer::Handle layer, synfig::String param_name)
+{
+ return canvas_interface()->change_value(synfigapp::ValueDesc(layer,param_name),value);
+}
+
+bool
+CanvasView::on_duck_changed(const synfig::Point &value,const synfigapp::ValueDesc& value_desc)
+{
+ if (ValueNode_BLineCalcWidth::Handle bline_width = ValueNode_BLineCalcWidth::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ Real old_width((*bline_width)(get_time()).get(Real()));
+ Real new_width(value.mag());
+ int scale_index(bline_width->get_link_index_from_name("scale"));
+ Real scale((*(bline_width->get_link(scale_index)))(get_time()).get(Real()));
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_width,scale_index), new_width * scale / old_width);
+ }
+
+ if (ValueNode_BLineCalcVertex::Handle bline_vertex = ValueNode_BLineCalcVertex::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ ValueNode_BLine::Handle bline = ValueNode_BLine::Handle::cast_dynamic(bline_vertex->get_link(bline_vertex->get_link_index_from_name("bline")));
+ Real radius = 0.0;
+ Real amount = synfig::find_closest_point((*bline)(get_time()), value, radius, bline->get_loop());
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_vertex,bline_vertex->get_link_index_from_name("amount")), amount);
+ }
+
+ if (ValueNode_BLineCalcTangent::Handle bline_tangent = ValueNode_BLineCalcTangent::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ switch(value_desc.get_value_type())
+ {
+ case ValueBase::TYPE_REAL:
+ {
+ Real old_length = (*bline_tangent)(get_time()).get(Real());
+ Real new_length = value.mag();
+ int scale_index(bline_tangent->get_link_index_from_name("scale"));
+ int fixed_length_index(bline_tangent->get_link_index_from_name("fixed_length"));
+ Real scale((*(bline_tangent->get_link(scale_index)))(get_time()).get(Real()));
+ bool fixed_length((*(bline_tangent->get_link(fixed_length_index)))(get_time()).get(bool()));
+ if (fixed_length)
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,scale_index), new_length);
+ if (old_length == 0)
+ return true;
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,scale_index), new_length * scale / old_length);
+ }
+
+ case ValueBase::TYPE_ANGLE:
+ assert(0); // doesn't happen?
+ break;
+
+ case ValueBase::TYPE_VECTOR:
+ {
+ Vector old_tangent = (*bline_tangent)(get_time()).get(Vector());
+ Angle old_angle = old_tangent.angle();
+ Real old_length = old_tangent.mag();
+ Angle new_angle = value.angle();
+ Real new_length = value.mag();
+ int offset_index(bline_tangent->get_link_index_from_name("offset"));
+ int scale_index(bline_tangent->get_link_index_from_name("scale"));
+ int fixed_length_index(bline_tangent->get_link_index_from_name("fixed_length"));
+ Angle old_offset((*(bline_tangent->get_link(offset_index)))(get_time()).get(Angle()));
+ Real scale((*(bline_tangent->get_link(scale_index)))(get_time()).get(Real()));
+ bool fixed_length((*(bline_tangent->get_link(fixed_length_index)))(get_time()).get(bool()));
+ if (fixed_length)
+ {
+ if (!(canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,scale_index), new_length)))
+ return false;
+ }
+ else if (old_length != 0 && !(canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,scale_index), new_length * scale / old_length)))
+ return false;
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,offset_index), old_offset + new_angle - old_angle);
+ }
+ default:
+ break;
+ }
+ }
+
+ if (ValueNode_Scale::Handle scale_value_node = ValueNode_Scale::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ int link_index(scale_value_node->get_link_index_from_name("link"));
+ if(scale_value_node->is_invertible(get_time()))
+ return canvas_interface()->change_value(
+ synfigapp::ValueDesc(scale_value_node,link_index),
+ scale_value_node->get_inverse(get_time(), value)
+ );
+ else
+ return false;
+ }
+
+ switch(value_desc.get_value_type())
+ {
+ case ValueBase::TYPE_REAL:
+ return canvas_interface()->change_value(value_desc,value.mag());
+ case ValueBase::TYPE_ANGLE:
+ return canvas_interface()->change_value(value_desc,Angle::tan(value[1],value[0]));
+ default:
+ return canvas_interface()->change_value(value_desc,value);
+ }
+}
+
+bool
+CanvasView::on_duck_angle_changed(const synfig::Angle &rotation,const synfigapp::ValueDesc& value_desc)
+{
+ if (ValueNode_BLineCalcTangent::Handle bline_tangent = ValueNode_BLineCalcTangent::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ int offset_index(bline_tangent->get_link_index_from_name("offset"));
+ Angle old_offset((*(bline_tangent->get_link(offset_index)))(get_time()).get(Angle()));
+ return canvas_interface()->change_value(synfigapp::ValueDesc(bline_tangent,offset_index), old_offset + rotation);
+ }
+
+ if (ValueNode_Scale::Handle scale_value_node = ValueNode_Scale::Handle::cast_dynamic(value_desc.get_value_node()))
+ {
+ int link_index(scale_value_node->get_link_index_from_name("link"));
+ if(scale_value_node->is_invertible(get_time()))
+ return canvas_interface()->change_value(
+ synfigapp::ValueDesc(scale_value_node,link_index),
+ scale_value_node->get_inverse(get_time(), rotation)
+ );
+ else
+ return false;
+
+ }
+ // \todo will this really always be the case?
+ assert(value_desc.get_value_type() == ValueBase::TYPE_ANGLE);
+ return canvas_interface()->change_value(value_desc, value_desc.get_value(get_time()).get(Angle()) + rotation);
+}
+
+void
+CanvasView::selected_layer_color_set(synfig::Color color)
+{
+ synfigapp::SelectionManager::LayerList selected_list(get_selection_manager()->get_selected_layers());
+ synfigapp::SelectionManager::LayerList::iterator iter;
+
+ // Create the action group
+ //synfigapp::PassiveGrouper group(canvas_interface()->get_instance(),_("Set Colors"));
+
+ Layer::Handle layer;
+ for(iter=selected_list.begin();iter!=selected_list.end();++iter)
+ {
+ if(*iter==layer)
+ continue;
+ layer=*iter;
+ on_edited_value(synfigapp::ValueDesc(layer,"color"),color);
+ }
+}
+
+void
+CanvasView::rebuild_ducks_layer_(synfig::TransformStack& transform_stack, synfig::Canvas::Handle canvas, std::set<synfig::Layer::Handle>& selected_list)
+{
+ int transforms(0);
+ String layer_name;
+
+#define QUEUE_REBUILD_DUCKS sigc::mem_fun(*this,&CanvasView::queue_rebuild_ducks)
+
+ if(!canvas)
+ {
+ synfig::warning("CanvasView::rebuild_ducks_layer_(): Layer doesn't have canvas set");
+ return;
+ }
+ for(Canvas::iterator iter(canvas->begin());iter!=canvas->end();++iter)
+ {
+ Layer::Handle layer(*iter);
+
+ if(selected_list.count(layer))
+ {
+ if(!curr_transform_stack_set)
+ {
+ curr_transform_stack_set=true;
+ curr_transform_stack=transform_stack;
+ }
+
+ // This layer is currently selected.
+ duck_changed_connections.push_back(layer->signal_changed().connect(QUEUE_REBUILD_DUCKS));
+
+ // do the bounding box thing
+ bbox|=transform_stack.perform(layer->get_bounding_rect());
+
+ // Grab the layer's list of parameters
+ Layer::ParamList paramlist(layer->get_param_list());
+
+ // Grab the layer vocabulary
+ Layer::Vocab vocab=layer->get_param_vocab();
+ Layer::Vocab::iterator iter;
+
+ for(iter=vocab.begin();iter!=vocab.end();iter++)
+ {
+ if(!iter->get_hidden() && !iter->get_invisible_duck())
+ {
+ synfigapp::ValueDesc value_desc(layer,iter->get_name());
+ work_area->add_to_ducks(value_desc,this,transform_stack,&*iter);
+ if(value_desc.is_value_node())
+ duck_changed_connections.push_back(value_desc.get_value_node()->signal_changed().connect(QUEUE_REBUILD_DUCKS));
+ }
+ if(iter->get_name()=="color")
+ {
+ /*
+ if(!App::dialog_color->busy())
+ {
+ App::dialog_color->reset();
+ App::dialog_color->set_color(layer->get_param("color").get(Color()));
+ App::dialog_color->signal_edited().connect(
+ sigc::mem_fun(
+ *this,
+ &studio::CanvasView::selected_layer_color_set
+ )
+ );
+ }
+ */
+ }
+ }
+ }
+
+ layer_name=layer->get_name();
+
+ if(layer->active())
+ {
+ Transform::Handle trans(layer->get_transform());
+ if(trans)
+ {
+ transform_stack.push(trans);
+ transforms++;
+ }
+
+/* // Add transforms onto the stack
+ if(layer_name=="Translate")
+ {
+ transform_stack.push(synfig::Transform_Translate(layer->get_param("origin").get(Vector())));
+ transforms++;
+ }else
+ if(layer_name=="Zoom")
+ {
+ Vector scale;
+ scale[0]=scale[1]=exp(layer->get_param("amount").get(Real()));
+ transform_stack.push(synfig::Transform_Scale(scale,layer->get_param("center").get(Vector())));
+ transforms++;
+ }else
+ if(layer_name=="stretch")
+ {
+ Vector scale(layer->get_param("amount").get(Vector()));
+ transform_stack.push(synfig::Transform_Scale(scale,layer->get_param("center").get(Vector())));
+ transforms++;
+ }else
+ if(layer_name=="Rotate")
+ {
+ transform_stack.push(synfig::Transform_Rotate(layer->get_param("amount").get(Angle()),layer->get_param("origin").get(Vector())));
+ transforms++;
+ }
+*/
+ }
+
+ // If this is a paste canvas layer, then we need to
+ // descend into it
+ if(layer_name=="PasteCanvas")
+ {
+ Vector scale;
+ scale[0]=scale[1]=exp(layer->get_param("zoom").get(Real()));
+ Vector origin(layer->get_param("origin").get(Vector()));
+
+ Canvas::Handle child_canvas(layer->get_param("canvas").get(Canvas::Handle()));
+ Vector focus(layer->get_param("focus").get(Vector()));
+
+ if(!scale.is_equal_to(Vector(1,1)))
+ transform_stack.push(new Transform_Scale(layer->get_guid(), scale,origin+focus));
+ if(!origin.is_equal_to(Vector(0,0)))
+ transform_stack.push(new Transform_Translate(layer->get_guid(), origin));
+
+ rebuild_ducks_layer_(transform_stack,child_canvas,selected_list);
+
+ if(!origin.is_equal_to(Vector(0,0)))
+ transform_stack.pop();
+ if(!scale.is_equal_to(Vector(1,1)))
+ transform_stack.pop();
+ }
+ }
+ // Remove all of the transforms we have added
+ while(transforms--) { transform_stack.pop(); }
+
+#undef QUEUE_REBUILD_DUCKS
+}
+
+void
+CanvasView::queue_rebuild_ducks()
+{
+#if 0
+ if(rebuild_ducks_queued)
+ return;
+#else
+ if(rebuild_ducks_queued)
+ queue_rebuild_ducks_connection.disconnect();
+#endif
+
+ queue_rebuild_ducks_connection=Glib::signal_timeout().connect(
+ sigc::bind_return(
+ sigc::mem_fun(*this,&CanvasView::rebuild_ducks),
+ false
+ ),
+ 50
+ );
+
+ rebuild_ducks_queued=true;
+}
+
+void
+CanvasView::rebuild_ducks()
+{
+ /*static int i=0;
+ i++;
+ if(i>30)
+ synfig::info("%d",i/(i-i));
+ */
+
+ rebuild_ducks_queued=false;
+ //queue_rebuild_ducks_connection.disconnect();
+
+ if(work_area->is_dragging())
+ {
+ queue_rebuild_ducks();
+ return;
+ }
+
+ if(!duck_refresh_flag)
+ {
+ duck_refresh_needed=true;
+ return;
+ }
+
+ bbox=Rect::zero();
+
+ work_area->clear_ducks();
+ work_area->set_time(get_time());
+ get_canvas()->set_time(get_time());
+ curr_transform_stack.clear();
+ //curr_transform_stack.push(new Transform_Translate(Point(0,0)));
+ curr_transform_stack_set=false;
+
+ for(;!duck_changed_connections.empty();duck_changed_connections.pop_back())duck_changed_connections.back().disconnect();
+
+ //get_canvas()->set_time(get_time());
+ bool not_empty(false);
+
+ // First do the layers...
+ do{
+ synfigapp::SelectionManager::LayerList selected_list(get_selection_manager()->get_selected_layers());
+ std::set<synfig::Layer::Handle> layer_set(selected_list.begin(),selected_list.end());
+
+ if(!layer_set.empty())
+ not_empty=true;
+
+ synfig::TransformStack transform_stack;
+
+ rebuild_ducks_layer_(transform_stack, get_canvas(), layer_set);
+
+ }while(0);
+
+ // Now do the children
+ do{
+ synfigapp::SelectionManager::ChildrenList selected_list(get_selection_manager()->get_selected_children());
+ synfigapp::SelectionManager::ChildrenList::iterator iter;
+ synfig::TransformStack transform_stack;
+
+ if(selected_list.empty())
+ {
+ break;
+ }
+ else
+ {
+ not_empty=true;
+ for(iter=selected_list.begin();iter!=selected_list.end();++iter)
+ {
+ work_area->add_to_ducks(*iter,this,transform_stack);
+ }
+ }
+ }while(0);
+ work_area->refresh_selected_ducks();
+ work_area->queue_draw_preview();
+}
+
+void
+CanvasView::decrease_low_res_pixel_size()
+{
+ if(changing_resolution_)
+ return;
+ changing_resolution_=true;
+ list<int> sizes = CanvasView::get_pixel_sizes();
+ int pixel_size = work_area->get_low_res_pixel_size();
+ for (list<int>::iterator iter = sizes.begin(); iter != sizes.end(); iter++)
+ if (*iter == pixel_size)
+ {
+ if (iter == sizes.begin())
+ // we already have the smallest low-res pixels possible - turn off low-res instead
+ work_area->set_low_resolution_flag(false);
+ else
+ {
+ iter--;
+ Glib::RefPtr<Gtk::Action> action = action_group->get_action(strprintf("lowres-pixel-%d", *iter));
+ action->activate(); // to make sure the radiobutton in the menu is updated too
+ work_area->set_low_resolution_flag(true);
+ }
+ break;
+ }
+ // Update the "toggle-low-res" action
+ Glib::RefPtr<Gtk::ToggleAction> action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("toggle-low-res"));
+ action->set_active(work_area->get_low_resolution_flag());
+ // Update toggle low res button
+ resolutiondial->update_lowres(work_area->get_low_resolution_flag());
+ changing_resolution_=false;
+}
+
+void
+CanvasView::increase_low_res_pixel_size()
+{
+ if(changing_resolution_)
+ return;
+ changing_resolution_=true;
+ list<int> sizes = CanvasView::get_pixel_sizes();
+ int pixel_size = work_area->get_low_res_pixel_size();
+ if (!work_area->get_low_resolution_flag())
+ {
+ // We were using "hi res" so change it to low res.
+ work_area->set_low_resolution_flag(true);
+ // Update the "toggle-low-res" action
+ Glib::RefPtr<Gtk::ToggleAction> action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("toggle-low-res"));
+ action->set_active(true);
+ // Update the toggle low res button
+ resolutiondial->update_lowres(true);
+ changing_resolution_=false;
+ return;
+ }
+
+ for (list<int>::iterator iter = sizes.begin(); iter != sizes.end(); iter++)
+ if (*iter == pixel_size)
+ {
+ iter++;
+ if (iter != sizes.end())
+ {
+ Glib::RefPtr<Gtk::Action> action = action_group->get_action(strprintf("lowres-pixel-%d", *iter));
+ action->activate(); // to make sure the radiobutton in the menu is updated too
+ work_area->set_low_resolution_flag(true);
+ }
+ break;
+ }
+ // Update the "toggle-low-res" action
+ Glib::RefPtr<Gtk::ToggleAction> action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("toggle-low-res"));
+ action->set_active(work_area->get_low_resolution_flag());
+ // Update toggle low res button
+ resolutiondial->update_lowres(work_area->get_low_resolution_flag());
+ changing_resolution_=false;
+}
+
+void
+CanvasView::toggle_low_res_pixel_flag()
+{
+ if(changing_resolution_)
+ return;
+ changing_resolution_=true;
+ work_area->toggle_low_resolution_flag();
+ // Update the toggle low res button
+ resolutiondial->update_lowres(work_area->get_low_resolution_flag());
+ // Update the "toggle-low-res" action
+ Glib::RefPtr<Gtk::ToggleAction> action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("toggle-low-res"));
+ action->set_active(work_area->get_low_resolution_flag());
+ changing_resolution_=false;
+}
+
+void
+CanvasView::update_quality()
+{
+ //if(working_depth)
+ // return;
+ if(updating_quality_)
+ return;
+ updating_quality_=true;
+ work_area->set_quality((int) quality_spin->get_value());
+ // Update Quality Radio actions
+ Glib::RefPtr<Gtk::RadioAction> action=Glib::RefPtr<Gtk::RadioAction>::cast_dynamic(
+ action_group->get_action(strprintf("quality-%02d",(int) quality_spin->get_value()))
+ );
+ action->set_active();
+
+ updating_quality_=false;
+}
+
+void
+CanvasView::set_quality(int x)
+{
+ if(updating_quality_)
+ return;
+ work_area->set_quality(x);
+ // Update the quality spin button
+ quality_spin->set_value(x);
+}
+
+void
+CanvasView::set_onion_skins()
+{
+ if(toggling_onion_skin)
+ return;
+ int onion_skins[2];
+ onion_skins[0]=past_onion_spin->get_value();
+ onion_skins[1]=future_onion_spin->get_value();
+ work_area->set_onion_skins(onion_skins);
+}
+
+void
+CanvasView::toggle_show_grid()
+{
+ if(toggling_show_grid)
+ return;
+ toggling_show_grid=true;
+ work_area->toggle_grid();
+ // Update the toggle grid show action
+ set_grid_show_toggle(work_area->grid_status());
+ // Update the toggle grid show check button
+ show_grid->set_active(work_area->grid_status());
+ toggling_show_grid=false;
+}
+
+void
+CanvasView::toggle_snap_grid()
+{
+ if(toggling_snap_grid)
+ return;
+ toggling_snap_grid=true;
+ work_area->toggle_grid_snap();
+ // Update the toggle grid snap action
+ set_grid_snap_toggle(work_area->get_grid_snap());
+ // Update the toggle grid snap check button
+ snap_grid->set_active(work_area->get_grid_snap());
+ toggling_snap_grid=false;
+}
+
+void
+CanvasView::toggle_onion_skin()
+{
+ if(toggling_onion_skin)
+ return;
+ toggling_onion_skin=true;
+ work_area->toggle_onion_skin();
+ // Update the toggle onion skin action
+ Glib::RefPtr<Gtk::ToggleAction> action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("toggle-onion-skin"));
+ action->set_active(work_area->get_onion_skin());
+ // Update the toggle grid snap check button
+ onion_skin->set_active(work_area->get_onion_skin());
+ toggling_onion_skin=false;
+}
+
+void
+CanvasView::on_dirty_preview()
+{
+ if(!is_playing_)
+ {
+ IsWorking is_working(*this);
+
+ work_area->queue_render_preview();
+ }
+}
+
+void
+CanvasView::play()
+{
+ assert(get_canvas());
+
+ // If we are already busy, don't play!
+ if(working_depth)return;
+
+ // Set us up as working
+ IsWorking is_working(*this);
+
+ etl::clock timer;
+ Time
+ time=work_area->get_time(),
+ endtime=get_canvas()->rend_desc().get_time_end();
+
+ // If we are already at the end of time, start over
+ if(time==endtime)
+ time=get_canvas()->rend_desc().get_time_start();
+
+ is_playing_=true;
+
+ work_area->clear_ducks();
+
+ for(timer.reset(); time + timer() < endtime;)
+ {
+ //Clamp the time window so we can see the time value as it races across the horizon
+ bool timewindreset = false;
+
+ while( time + timer() > Time(time_window_adjustment().get_sub_upper()) )
+ {
+ time_window_adjustment().set_value(
+ min(
+ time_window_adjustment().get_value()+time_window_adjustment().get_page_size()/2,
+ time_window_adjustment().get_upper()-time_window_adjustment().get_page_size() )
+ );
+ timewindreset = true;
+ }
+
+ while( time + timer() < Time(time_window_adjustment().get_sub_lower()) )
+ {
+ time_window_adjustment().set_value(
+ max(
+ time_window_adjustment().get_value()-time_window_adjustment().get_page_size()/2,
+ time_window_adjustment().get_lower())
+ );
+
+ timewindreset = true;
+ }
+
+ //we need to tell people that the value changed
+ if(timewindreset) time_window_adjustment().value_changed();
+
+ //update actual time to next step
+ time_adjustment().set_value(time+timer());
+ time_adjustment().value_changed();
+
+ if(!work_area->sync_render_preview())
+ break;
+
+ // wait for the workarea to refresh itself
+ while (studio::App::events_pending())
+ studio::App::iteration(false);
+
+ if(get_cancel_status())
+ {
+ is_playing_=false;
+ return;
+ }
+ }
+ on_play_stop_pressed();
+ is_playing_=false;
+ time_adjustment().set_value(endtime);
+ time_adjustment().value_changed();
+}
+
+void
+CanvasView::show_tables()
+{
+/*
+ Smach::event_result x(process_event_key(EVENT_TABLES_SHOW));
+ if(x==Smach::RESULT_OK || x==Smach::RESULT_ACCEPT)
+ {
+ Gtk::IconSize iconsize=Gtk::IconSize::from_name("synfig-small_icon");
+ treetogglebutton->remove();
+ treetogglebutton->add(*manage(new Gtk::Image(Gtk::StockID("gtk-go-down"),iconsize)));
+ treetogglebutton->show_all();
+ notebook->show();
+ }
+*/
+}
+
+void
+CanvasView::hide_tables()
+{
+/*
+ Smach::event_result x(process_event_key(EVENT_TABLES_HIDE));
+ if(x==Smach::RESULT_OK || x==Smach::RESULT_ACCEPT)
+ {
+ Gtk::IconSize iconsize=Gtk::IconSize::from_name("synfig-small_icon");
+ treetogglebutton->remove();
+ treetogglebutton->add(*manage(new Gtk::Image(Gtk::StockID("gtk-go-up"),iconsize)));
+ treetogglebutton->show_all();
+ notebook->hide();
+ }
+*/
+}
+
+bool
+CanvasView::tables_are_visible()
+{
+// return notebook->is_visible();
+ return false;
+}
+
+void
+CanvasView::toggle_tables()
+{
+// if(tables_are_visible())
+// hide_tables();
+// else
+// show_tables();
+}
+
+void
+CanvasView::show_timebar()
+{
+ timebar->show();
+ //current_time_widget->show(); // not needed now that belongs to the timebar
+
+ //keyframe_tab_child->show();
+ if(layer_tree)
+ layer_tree->set_show_timetrack(true);
+ if(children_tree)
+ children_tree->set_show_timetrack(true);
+}
+
+void
+CanvasView::hide_timebar()
+{
+ timebar->hide();
+ //current_time_widget->hide(); // not needed now that belongs to the timebar
+ //keyframe_tab_child->hide();
+ if(layer_tree)
+ layer_tree->set_show_timetrack(false);
+ if(children_tree)
+ children_tree->set_show_timetrack(false);
+}
+
+void
+CanvasView::set_sensitive_timebar(bool sensitive)
+{
+ timebar->set_sensitive(sensitive);
+ //current_time_widget->set_sensitive(sensitive); //not needed now that belongs to timebar
+ //keyframe_tab_child->set_sensitive(sensitive);
+ if(layer_tree)
+ layer_tree->set_sensitive(sensitive);
+ if(children_tree)
+ children_tree->set_sensitive(sensitive);
+}
+
+static void
+set_waypoint_model(std::set<synfig::Waypoint, std::less<UniqueID> > waypoints,
+ Waypoint::Model model,
+ etl::loose_handle<synfigapp::CanvasInterface> canvas_interface)
+{
+ // Create the action group
+ synfigapp::Action::PassiveGrouper group(canvas_interface->get_instance().get(),_("Change Waypoint Group"));
+
+ std::set<synfig::Waypoint, std::less<UniqueID> >::const_iterator iter;
+ for(iter=waypoints.begin();iter!=waypoints.end();++iter)
+ {
+ Waypoint waypoint(*iter);
+ waypoint.apply_model(model);
+
+ synfigapp::Action::Handle action(synfigapp::Action::create("WaypointSet"));
+
+ assert(action);
+
+ action->set_param("canvas",canvas_interface->get_canvas());
+ action->set_param("canvas_interface",canvas_interface);
+
+ action->set_param("waypoint",waypoint);
+ action->set_param("value_node",waypoint.get_parent_value_node());
+
+ if(!canvas_interface->get_instance()->perform_action(action))
+ {
+ group.cancel();
+ return;
+ }
+ }
+}
+
+static void
+duplicate_waypoints(std::set<synfig::Waypoint, std::less<UniqueID> > waypoints,
+ etl::loose_handle<synfigapp::CanvasInterface> canvas_interface)
+{
+ // Create the action group
+ synfigapp::Action::PassiveGrouper group(canvas_interface->get_instance().get(),_("Duplicate Waypoints"));
+
+ std::set<synfig::Waypoint, std::less<UniqueID> >::const_iterator iter;
+ for (iter = waypoints.begin(); iter != waypoints.end(); iter++)
+ {
+ Waypoint waypoint(*iter);
+ ValueNode::Handle value_node(iter->get_parent_value_node());
+ canvas_interface->waypoint_duplicate(value_node, waypoint);
+ }
+}
+
+static void
+remove_waypoints(std::set<synfig::Waypoint, std::less<UniqueID> > waypoints,
+ etl::loose_handle<synfigapp::CanvasInterface> canvas_interface)
+{
+ // Create the action group
+ synfigapp::Action::PassiveGrouper group(canvas_interface->get_instance().get(),_("Remove Waypoints"));
+
+ std::set<synfig::Waypoint, std::less<UniqueID> >::const_iterator iter;
+ for (iter = waypoints.begin(); iter != waypoints.end(); iter++)
+ {
+ Waypoint waypoint(*iter);
+ ValueNode::Handle value_node(iter->get_parent_value_node());
+ canvas_interface->waypoint_remove(value_node, waypoint);
+ }
+}
+
+void
+CanvasView::on_waypoint_clicked_canvasview(synfigapp::ValueDesc value_desc,
+ std::set<synfig::Waypoint, std::less<UniqueID> > waypoint_set,
+ int button)
+{
+ int size = waypoint_set.size();
+ Waypoint waypoint(*(waypoint_set.begin()));
+ Time time(waypoint.get_time());
+
+ if (size == 1)
+ {
+ waypoint_dialog.set_value_desc(value_desc);
+ waypoint_dialog.set_waypoint(waypoint);
+ }
+
+ switch(button)
+ {
+ case -1:
+ if (size == 1)
+ waypoint_dialog.show();
+ break;
+ case 2:
+ {
+ Gtk::Menu* waypoint_menu(manage(new Gtk::Menu()));
+ waypoint_menu->signal_hide().connect(sigc::bind(sigc::ptr_fun(&delete_widget), waypoint_menu));
+
+ Gtk::Menu* interp_menu_in(manage(new Gtk::Menu()));
+ Gtk::Menu* interp_menu_out(manage(new Gtk::Menu()));
+ Gtk::Menu* interp_menu_both(manage(new Gtk::Menu()));
+
+ {
+ Waypoint::Model model;
+
+ model.reset(); model.set_before(INTERPOLATION_TCB);
+ interp_menu_in->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_TCB"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.reset(); model.set_after(INTERPOLATION_TCB);
+ interp_menu_out->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_TCB"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.set_before(INTERPOLATION_TCB);
+ interp_menu_both->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_TCB"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+
+ model.reset(); model.set_before(INTERPOLATION_LINEAR);
+ interp_menu_in->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Linear"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.reset(); model.set_after(INTERPOLATION_LINEAR);
+ interp_menu_out->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Linear"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.set_before(INTERPOLATION_LINEAR);
+ interp_menu_both->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Linear"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+
+ model.reset(); model.set_before(INTERPOLATION_HALT);
+ interp_menu_in->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Ease In"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.reset(); model.set_after(INTERPOLATION_HALT);
+ interp_menu_out->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Ease Out"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.set_before(INTERPOLATION_HALT);
+ interp_menu_both->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Ease In/Out"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+
+ model.reset(); model.set_before(INTERPOLATION_CONSTANT);
+ interp_menu_in->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Constant"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.reset(); model.set_after(INTERPOLATION_CONSTANT);
+ interp_menu_out->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Constant"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ model.set_before(INTERPOLATION_CONSTANT);
+ interp_menu_both->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Constant"),
+ sigc::bind(sigc::ptr_fun(set_waypoint_model), waypoint_set, model, canvas_interface())));
+ }
+
+ // ------------------------------------------------------------------------
+ if (size == 1)
+ {
+ const synfigapp::ValueDesc value_desc(synfig::ValueNode_Animated::Handle::cast_reinterpret(waypoint.get_parent_value_node()), time);
+ get_instance()->make_param_menu(waypoint_menu,canvas_interface()->get_canvas(),value_desc,0.5f);
+
+ // ------------------------------------------------------------------------
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+ }
+
+ // ------------------------------------------------------------------------
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Jump To"),
+ sigc::bind(sigc::mem_fun(*canvas_interface(), &synfigapp::CanvasInterface::set_time), time)));
+
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Duplicate"),
+ sigc::bind(sigc::ptr_fun(duplicate_waypoints), waypoint_set, canvas_interface())));
+
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem((size == 1) ? _("_Remove") : strprintf(_("_Remove %d Waypoints"), size),
+ sigc::bind(sigc::ptr_fun(remove_waypoints), waypoint_set, canvas_interface())));
+
+ if (size == 1 && value_desc.is_valid())
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Edit"),
+ sigc::mem_fun(waypoint_dialog,&Gtk::Widget::show)));
+
+ // ------------------------------------------------------------------------
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::SeparatorElem());
+
+ // ------------------------------------------------------------------------
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Both"), *interp_menu_both));
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_In"), *interp_menu_in));
+ waypoint_menu->items().push_back(Gtk::Menu_Helpers::MenuElem(_("_Out"), *interp_menu_out));
+
+ // ------------------------------------------------------------------------
+ waypoint_menu->popup(button+1,gtk_get_current_event_time());
+ }
+ break;
+
+ default:
+ break;
+ }
+}
+
+void
+CanvasView::on_waypoint_changed()
+{
+ synfigapp::Action::ParamList param_list;
+ param_list.add("canvas",get_canvas());
+ param_list.add("canvas_interface",canvas_interface());
+ param_list.add("value_node",waypoint_dialog.get_value_desc().get_value_node());
+ param_list.add("waypoint",waypoint_dialog.get_waypoint());
+// param_list.add("time",canvas_interface()->get_time());
+
+ get_instance()->process_action("WaypointSetSmart", param_list);
+}
+
+void
+CanvasView::on_waypoint_delete()
+{
+ synfigapp::Action::ParamList param_list;
+ param_list.add("canvas",get_canvas());
+ param_list.add("canvas_interface",canvas_interface());
+ param_list.add("value_node",waypoint_dialog.get_value_desc().get_value_node());
+ param_list.add("waypoint",waypoint_dialog.get_waypoint());
+// param_list.add("time",canvas_interface()->get_time());
+
+ get_instance()->process_action("WaypointRemove", param_list);
+}
+
+void
+CanvasView::on_drop_drag_data_received(const Glib::RefPtr<Gdk::DragContext>& context, int /*x*/, int /*y*/, const Gtk::SelectionData& selection_data_, guint /*info*/, guint time)
+{
+ // We will make this true once we have a solid drop
+ bool success(false);
+ //synfig::info("Dropped data of type \"%s\"",selection_data.get_data_type());
+ //synfig::info("Dropped data of target \"%s\"",gdk_atom_name(selection_data->target));
+ //synfig::info("selection=\"%s\"",gdk_atom_name(selection_data->selection));
+
+ if ((selection_data_.get_length() >= 0) && (selection_data_.get_format() == 8))
+ {
+ if(synfig::String(selection_data_.get_data_type())=="STRING")do
+ {
+ synfig::String selection_data((gchar *)(selection_data_.get_data()));
+
+ Layer::Handle layer(synfig::Layer::create("Text"));
+ if(!layer)
+ break;
+ if(!layer->set_param("text",ValueBase(selection_data)))
+ break;
+
+ synfigapp::Action::Handle action(synfigapp::Action::create("LayerAdd"));
+
+ assert(action);
+ if(!action)
+ break;
+
+ action->set_param("canvas",get_canvas());
+ action->set_param("canvas_interface",canvas_interface());
+ action->set_param("new",layer);
+
+ if(!get_instance()->perform_action(action))
+ break;
+
+ // Ok, we have successfully imported at least one item.
+ success=true;
+ } while(0); // END of "STRING"
+
+ if(synfig::String(selection_data_.get_data_type())=="text/plain")
+ {
+ synfig::String selection_data((gchar *)(selection_data_.get_data()));
+
+ // For some reason, GTK hands us a list of URLs separated
+ // by not only Carriage-Returns, but also Line-Feeds.
+ // Line-Feeds will mess us up. Remove all the line-feeds.
+ while(selection_data.find_first_of('\r')!=synfig::String::npos)
+ selection_data.erase(selection_data.begin()+selection_data.find_first_of('\r'));
+
+ std::stringstream stream(selection_data);
+
+ //synfigapp::PassiveGrouper group(canvas_interface()->get_instance(),_("Insert Image"));
+ while(stream)
+ {
+ synfig::String filename,URI;
+ getline(stream,filename);
+
+ // If we don't have a filename, move on.
+ if(filename.empty())
+ continue;
+
+ // Make sure this URL is of the "file://" type.
+ URI=String(filename.begin(),filename.begin()+sizeof("file://")-1);
+ if(URI!="file://")
+ {
+ synfig::warning("Unknown URI (%s) in \"%s\"",URI.c_str(),filename.c_str());
+ continue;
+ }
+
+ // Strip the "file://" part from the filename
+ filename=synfig::String(filename.begin()+sizeof("file://")-1,filename.end());
+
+ String ext(filename_extension(filename));
+ if (ext.size()) ext = ext.substr(1); // skip initial '.'
+
+ // If this is a SIF file, then we need to do things slightly differently
+ if(ext=="sketch")
+ {
+ if(work_area->load_sketch(filename))
+ {
+ success=true;
+ work_area->queue_draw();
+ }
+ }
+ else
+ {
+ String errors, warnings;
+ if(canvas_interface()->import(filename, errors, warnings, App::resize_imported_images))
+ success=true;
+ if (warnings != "")
+ App::dialog_warning_blocking(_("Warnings"), strprintf("%s:\n\n%s", _("Warnings"), warnings.c_str()));
+ }
+
+ continue;
+ }
+ } // END of "text/plain"
+ }
+ else
+ ui_interface_->error("Drop failed: bad selection data");
+
+ // Finish the drag
+ context->drag_finish(success, false, time);
+}
+
+void
+CanvasView::on_keyframe_add_pressed()
+{
+ synfigapp::Action::Handle action(synfigapp::Action::create("KeyframeAdd"));
+
+ if(!action)
+ {
+ ui_interface_->error("I am unable to find the appropriate action");
+ return;
+ }
+
+ action->set_param("canvas",get_canvas());
+ action->set_param("canvas_interface",canvas_interface());
+ action->set_param("keyframe",Keyframe(get_time()));
+
+ canvas_interface()->get_instance()->perform_action(action);
+}
+
+void
+CanvasView::on_keyframe_duplicate_pressed()
+{
+ const KeyframeTreeStore::Model model;
+ const Gtk::TreeRow row(*keyframe_tree->get_selection()->get_selected());
+ Keyframe keyframe;
+ if(!row)
+ {
+ ui_interface_->error("I am unable to duplicate the keyframe");
+ return;
+ }
+ keyframe=row[model.keyframe];
+
+ synfigapp::Action::Handle action(synfigapp::Action::create("KeyframeDuplicate"));
+
+ if(!action)
+ {
+ ui_interface_->error("I am unable to find the appropriate action");
+ return;
+ }
+
+ action->set_param("canvas",get_canvas());
+ action->set_param("canvas_interface",canvas_interface());
+ action->set_param("keyframe",keyframe);
+ action->set_param("time",get_time());
+
+ canvas_interface()->get_instance()->perform_action(action);
+}
+
+void
+CanvasView::on_keyframe_remove_pressed()
+{
+ const KeyframeTreeStore::Model model;
+ const Gtk::TreeRow row(*keyframe_tree->get_selection()->get_selected());
+ Keyframe keyframe;
+ if(!row)
+ {
+ ui_interface_->error("I am unable to remove the keyframe");
+ return;
+ }
+ keyframe=row[model.keyframe];
+
+ synfigapp::Action::Handle action(synfigapp::Action::create("KeyframeRemove"));
+
+ if(!action)
+ {
+ ui_interface_->error("I am unable to find the appropriate action");
+ return;
+ }
+
+ action->set_param("canvas",get_canvas());
+ action->set_param("canvas_interface",canvas_interface());
+ action->set_param("keyframe",keyframe);
+
+ canvas_interface()->get_instance()->perform_action(action);
+}
+
+void
+CanvasView::toggle_duck_mask(Duckmatic::Type type)
+{
+ if(toggling_ducks_)
+ return;
+ toggling_ducks_=true;
+ bool is_currently_on(work_area->get_type_mask()&type);
+
+ if(is_currently_on)
+ work_area->set_type_mask(work_area->get_type_mask()-type);
+ else
+ work_area->set_type_mask(work_area->get_type_mask()|type);
+
+ work_area->queue_draw();
+ try
+ {
+ // Update the toggle ducks actions
+ Glib::RefPtr<Gtk::ToggleAction> action;
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-position-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_POSITION));
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-tangent-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_TANGENT));
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-vertex-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_VERTEX));
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-radius-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_RADIUS));
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-width-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_WIDTH));
+ action = Glib::RefPtr<Gtk::ToggleAction>::cast_dynamic(action_group->get_action("mask-angle-ducks"));
+ action->set_active((bool)(work_area->get_type_mask()&Duck::TYPE_ANGLE));
+ // Update toggle ducks buttons
+ toggleducksdial->update_toggles(work_area->get_type_mask());
+ }
+ catch(...)
+ {
+ toggling_ducks_=false;
+ }
+ toggling_ducks_=false;
+}
+
+void
+CanvasView::image_import()
+{
+ // String filename(dirname(get_canvas()->get_file_name()));
+ String filename("*.*");
+ String errors, warnings;
+ if(App::dialog_open_file(_("Import Image"), filename, IMAGE_DIR_PREFERENCE))
+ {
+ canvas_interface()->import(filename, errors, warnings, App::resize_imported_images);
+ if (warnings != "")
+ App::dialog_warning_blocking(_("Warnings"), strprintf("%s:\n\n%s", _("Warnings"), warnings.c_str()));
+ }
+}
+
+Smach::event_result
+CanvasView::process_event_key(EventKey x)
+{
+ return smach_.process_event(x);
+}
+
+void
+CanvasView::on_input_device_changed(GdkDevice* device)
+{
+ if(!device)
+ {
+ get_smach().egress();
+ }
+ assert(device);
+
+ synfigapp::InputDevice::Handle input_device;
+ input_device=synfigapp::Main::select_input_device(device->name);
+ App::toolbox->change_state(input_device->get_state());
+ process_event_key(EVENT_INPUT_DEVICE_CHANGED);
+}
+
+void
+CanvasView::on_preview_option()
+{
+ Dialog_PreviewOptions *po = dynamic_cast<Dialog_PreviewOptions *>(get_ext_widget("prevoptions"));
+
+ Canvas::Handle canv = get_canvas();
+
+ if(canv)
+ {
+ RendDesc &r = canv->rend_desc();
+ if(r.get_frame_rate())
+ {
+ float rate = 1/r.get_frame_rate();
+ float beg = r.get_time_start() + r.get_frame_start()*rate;
+ float end = r.get_time_start() + r.get_frame_end()*rate;
+
+ if(!po)
+ {
+ po = new Dialog_PreviewOptions;
+ po->set_zoom(work_area->get_zoom()/2);
+ po->set_fps(r.get_frame_rate()/2);
+ po->set_begintime(beg);
+ po->set_begin_override(false);
+ po->set_endtime(end);
+ po->set_end_override(false);
+
+ set_ext_widget("prevoptions",po);
+ }
+ /*po->set_zoom(work_area->get_zoom()/2);
+ po->set_fps(r.get_frame_rate()/2);
+ po->set_begintime(beg);
+ po->set_begin_override(false);
+ po->set_endtime(end);
+ po->set_end_override(false);*/
+
+ po->set_global_fps(r.get_frame_rate());
+ po->signal_finish().connect(sigc::mem_fun(*this,&CanvasView::on_preview_create));
+ po->present();
+ }
+ }
+}
+
+void
+CanvasView::on_preview_create(const PreviewInfo &info)
+{
+ //set all the options
+ etl::handle<Preview> prev = new Preview;
+
+ prev->set_canvasview(this);
+ prev->set_zoom(info.zoom);
+ prev->set_fps(info.fps);
+ prev->set_overbegin(info.overbegin);
+ prev->set_begintime(info.begintime);
+ prev->set_overend(info.overend);
+ prev->set_endtime(info.endtime);
+ prev->set_quality(work_area->get_quality());
+
+ //render it out...
+ prev->render();
+
+ Dialog_Preview *pd = preview_dialog.get();
+ assert(pd);
+
+ pd->set_default_size(700,510);
+ pd->set_preview(prev.get());
+ pd->present();
+}
+
+void
+CanvasView::on_audio_option()
+{
+ synfig::warning("Launching Audio Options");
+ sound_dialog->set_global_fps(get_canvas()->rend_desc().get_frame_rate());
+ sound_dialog->present();
+}
+
+void
+CanvasView::on_audio_file_change(const std::string &f)
+{
+ //save in meta data - always even when not valid...
+ canvas_interface()->set_meta_data("audiofile",f);
+}
+
+void
+CanvasView::on_audio_offset_change(const synfig::Time &t)
+{
+ canvas_interface()->set_meta_data("audiooffset",t.get_string());
+}
+
+void
+CanvasView::on_audio_file_notify()
+{
+ std::string file(get_canvas()->get_meta_data("audiofile"));
+ if(!file.c_str()) return;
+
+ if(!audio->load(file,dirname(get_canvas()->get_file_name())+string("/")))
+ {
+ if(file != "") synfig::warning("Could not load the file: %s", file.c_str());
+ get_canvas()->erase_meta_data("audiofile");
+ disp_audio->hide();
+ disp_audio->set_profile(etl::handle<AudioProfile>());
+ }else
+ {
+ //save in canvasview
+ synfig::warning("Getting the profile of the music stuff");
+
+ //profile specific stuff for the preview widget
+ //similar for other attachments
+ Dialog_Preview *pd = preview_dialog.get();
+ pd->get_widget().set_audio(audio);
+
+ handle<AudioProfile> prof = audio->get_profile();
+
+ if(!prof)
+ {
+ synfig::warning("Agh, I couldn't build the profile captain!");
+ }
+ pd->get_widget().set_audioprofile(prof);
+
+ disp_audio->set_profile(audio->get_profile());
+ disp_audio->show();
+
+ synfig::warning("successfully set the profiles and stuff");
+ }
+ disp_audio->queue_draw();
+}
+
+void
+CanvasView::on_audio_offset_notify()
+{
+ Time t(get_canvas()->get_meta_data("audiooffset"),get_canvas()->rend_desc().get_frame_rate());
+ audio->set_offset(t);
+ sound_dialog->set_offset(t);
+ disp_audio->queue_draw();
+
+ // synfig::info("CanvasView::on_audio_offset_notify(): offset time set to %s",t.get_string(get_canvas()->rend_desc().get_frame_rate()).c_str());
+}
+
+void
+CanvasView::play_audio(float t)
+{
+ if(audio.get())
+ {
+ synfig::info("Playing audio at %f s",t);
+ audio->play(t);
+ }
+}
+
+void
+CanvasView::stop_audio()
+{
+ if(audio.get())
+ {
+ audio->stop();
+ }
+}
+
+bool
+CanvasView::on_audio_scrub()
+{
+ disp_audio->draw();
+ return true;
+}
+
+Glib::RefPtr<Glib::ObjectBase>
+CanvasView::get_ref_obj(const synfig::String& x)
+{
+ return ref_obj_book_[x];
+}
+
+Glib::RefPtr<const Glib::ObjectBase>
+CanvasView::get_ref_obj(const synfig::String& x)const
+{
+ return ref_obj_book_.find(x)->second;
+}
+
+void
+CanvasView::set_ref_obj(const synfig::String& x, Glib::RefPtr<Glib::ObjectBase> y)
+{
+ ref_obj_book_[x]=y;
+}
+
+Glib::RefPtr<Gtk::TreeModel>
+CanvasView::get_tree_model(const synfig::String& x)
+{
+ return Glib::RefPtr<Gtk::TreeModel>::cast_dynamic(ref_obj_book_["_tree_model_"+x]);
+}
+
+Glib::RefPtr<const Gtk::TreeModel>
+CanvasView::get_tree_model(const synfig::String& x)const
+{
+ return Glib::RefPtr<Gtk::TreeModel>::cast_dynamic(ref_obj_book_.find("_tree_model_"+x)->second);
+}
+
+void
+CanvasView::set_tree_model(const synfig::String& x, Glib::RefPtr<Gtk::TreeModel> y)
+{
+ ref_obj_book_["_tree_model_"+x]=Glib::RefPtr<Glib::ObjectBase>::cast_static(y);
+}
+
+Gtk::Widget*
+CanvasView::get_ext_widget(const synfig::String& x)
+{
+ return ext_widget_book_[x];
+}
+
+void
+CanvasView::set_ext_widget(const synfig::String& x, Gtk::Widget* y)
+{
+ ext_widget_book_[x]=y;
+ if(x=="layers_cmp")
+ {
+ layer_tree=dynamic_cast<LayerTree*>(y);
+
+ layer_tree->get_selection()->signal_changed().connect(SLOT_EVENT(EVENT_LAYER_SELECTION_CHANGED));
+ layer_tree->get_selection()->signal_changed().connect(SLOT_EVENT(EVENT_REFRESH_DUCKS));
+ layer_tree->signal_layer_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_layer_user_click));
+ layer_tree->signal_param_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_children_user_click));
+ layer_tree->signal_waypoint_clicked_layertree().connect(sigc::mem_fun(*this, &studio::CanvasView::on_waypoint_clicked_canvasview));
+ }
+ if(x=="children")
+ {
+ children_tree=dynamic_cast<ChildrenTree*>(y);
+ if(children_tree)children_tree->signal_user_click().connect(sigc::mem_fun(*this, &studio::CanvasView::on_children_user_click));
+ if(children_tree)children_tree->signal_waypoint_clicked_childrentree().connect(sigc::mem_fun(*this, &studio::CanvasView::on_waypoint_clicked_canvasview));
+ if(children_tree)children_tree->get_selection()->signal_changed().connect(SLOT_EVENT(EVENT_REFRESH_DUCKS));
+ }
+ if(x=="keyframes")
+ keyframe_tree=dynamic_cast<KeyframeTree*>(y);
+}
+
+bool
+CanvasView::on_delete_event(GdkEventAny* event __attribute__ ((unused)))
+{
+ close_view();
+
+ //! \todo This causes the window to be deleted straight away - but what if we prompt 'save?' and the user cancels?
+ // Is there ever any need to pass on the delete event to the window here?
+ // if(event) return Gtk::Window::on_delete_event(event);
+
+ return true;
+}
+
+//! Modify the play stop button apearence and play stop the animation
+void
+CanvasView::on_play_stop_pressed()
+{
+ Gtk::Image *icon;
+ Gtk::Button *stop_button;
+ stop_button=framedial->get_play_button();
+ bool play_flag;
+ if(!is_playing())
+ {
+ icon = manage(new Gtk::Image(Gtk::Stock::MEDIA_STOP, Gtk::IconSize::from_name("synfig-small_icon")));
+ stop_button->set_relief(Gtk::RELIEF_NORMAL);
+ play_flag=true;
+ }
+ else
+ {
+ icon = manage(new Gtk::Image(Gtk::Stock::MEDIA_PLAY, Gtk::IconSize::from_name("synfig-small_icon")));
+ stop_button->set_relief(Gtk::RELIEF_NONE);
+ play_flag=false;
+ }
+ stop_button->remove();
+ stop_button->add(*icon);
+ icon->set_padding(0, 0);
+ icon->show();
+ if(play_flag) play(); else stop();
+}