Show the 'select all child layers' icon in the layers panel all the time, rather...
[synfig.git] / synfig-studio / trunk / src / gtkmm / layeractionmanager.cpp
1 /* === S Y N F I G ========================================================= */
2 /*!     \file layeractionmanager.cpp
3 **      \brief Template File
4 **
5 **      $Id$
6 **
7 **      \legal
8 **      Copyright (c) 2002-2005 Robert B. Quattlebaum Jr., Adrian Bentley
9 **      Copyright (c) 2007, 2008 Chris Moore
10 **
11 **      This package is free software; you can redistribute it and/or
12 **      modify it under the terms of the GNU General Public License as
13 **      published by the Free Software Foundation; either version 2 of
14 **      the License, or (at your option) any later version.
15 **
16 **      This package is distributed in the hope that it will be useful,
17 **      but WITHOUT ANY WARRANTY; without even the implied warranty of
18 **      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
19 **      General Public License for more details.
20 **      \endlegal
21 */
22 /* ========================================================================= */
23
24 /* === H E A D E R S ======================================================= */
25
26 #ifdef USING_PCH
27 #       include "pch.h"
28 #else
29 #ifdef HAVE_CONFIG_H
30 #       include <config.h>
31 #endif
32
33 #include "layeractionmanager.h"
34 #include "layertree.h"
35 #include <synfig/context.h>
36 #include <synfigapp/action_param.h>
37 #include "instance.h"
38 #include <synfigapp/selectionmanager.h>
39
40 #include "general.h"
41
42 #endif
43
44 /* === U S I N G =========================================================== */
45
46 using namespace std;
47 using namespace etl;
48 using namespace synfig;
49 using namespace studio;
50
51 static const guint no_prev_popup((guint)-1);
52
53 /* === M A C R O S ========================================================= */
54
55 //#define ONE_ACTION_GROUP 1
56
57 /* === G L O B A L S ======================================================= */
58
59 /* === P R O C E D U R E S ================================================= */
60
61 /* === M E T H O D S ======================================================= */
62
63 LayerActionManager::LayerActionManager():
64         action_group_(Gtk::ActionGroup::create()),
65         popup_id_(no_prev_popup),
66         action_group_copy_paste(Gtk::ActionGroup::create()),
67         queued(false)
68 {
69         action_cut_=Gtk::Action::create(
70                 "cut",
71                 Gtk::StockID("gtk-cut")
72         );
73         action_cut_->signal_activate().connect(
74                 sigc::mem_fun(
75                         *this,
76                         &LayerActionManager::cut
77                 )
78         );
79         action_copy_=Gtk::Action::create(
80                 "copy",
81                 Gtk::StockID("gtk-copy")
82         );
83         action_copy_->signal_activate().connect(
84                 sigc::mem_fun(
85                         *this,
86                         &LayerActionManager::copy
87                 )
88         );
89         action_paste_=Gtk::Action::create(
90                 "paste",
91                 Gtk::StockID("gtk-paste")
92         );
93         action_paste_->signal_activate().connect(
94                 sigc::mem_fun(
95                         *this,
96                         &LayerActionManager::paste
97                 )
98         );
99
100
101         action_amount_inc_=Gtk::Action::create(
102                 "amount-inc",
103                 Gtk::StockID("gtk-add"),
104                 _("Increase Amount"),_("Increase Amount")
105         );
106         action_amount_inc_->signal_activate().connect(
107                 sigc::mem_fun(
108                         *this,
109                         &LayerActionManager::amount_inc
110                 )
111         );
112
113         action_amount_dec_=Gtk::Action::create(
114                 "amount-dec",
115                 Gtk::StockID("gtk-remove"),
116                 _("Decrease Amount"),_("Decrease Amount")
117         );
118         action_amount_dec_->signal_activate().connect(
119                 sigc::mem_fun(
120                         *this,
121                         &LayerActionManager::amount_dec
122                 )
123         );
124
125         action_amount_=Gtk::Action::create(
126                 "amount",
127                 Gtk::StockID("gtk-index"),
128                 _("Amount"),_("Amount")
129         );
130
131         action_select_all_child_layers_=Gtk::Action::create(
132                 "select-all-child-layers",
133                 Gtk::StockID("synfig-select_all_child_layers"),
134                 _("Select All Child Layers"),_("Select All Child Layers")
135         );
136         action_select_all_child_layers_->set_sensitive(false);
137 }
138
139 LayerActionManager::~LayerActionManager()
140 {
141 }
142
143 void
144 LayerActionManager::set_ui_manager(const Glib::RefPtr<Gtk::UIManager> &x)
145 {
146         clear();
147
148 #ifdef ONE_ACTION_GROUP
149         if(ui_manager_) get_ui_manager()->remove_action_group(action_group_);
150         ui_manager_=x;
151         if(ui_manager_) get_ui_manager()->insert_action_group(action_group_);
152 #else
153         ui_manager_=x;
154 #endif
155 }
156
157 void
158 LayerActionManager::set_layer_tree(LayerTree* x)
159 {
160         selection_changed_connection.disconnect();
161         layer_tree_=x;
162         if(layer_tree_)
163         {
164                 selection_changed_connection=layer_tree_->get_selection()->signal_changed().connect(
165                         sigc::mem_fun(*this,&LayerActionManager::queue_refresh)
166                 );
167         }
168 }
169
170 void
171 LayerActionManager::set_canvas_interface(const etl::handle<synfigapp::CanvasInterface> &x)
172 {
173         canvas_interface_=x;
174 }
175
176 void
177 LayerActionManager::clear()
178 {
179         if(ui_manager_)
180         {
181                 // Clear out old stuff
182                 if(popup_id_!=no_prev_popup)
183                 {
184                         get_ui_manager()->remove_ui(popup_id_);
185                         if(action_group_)get_ui_manager()->ensure_update();
186                         popup_id_=no_prev_popup;
187                         if(action_group_)while(!action_group_->get_actions().empty())action_group_->remove(*action_group_->get_actions().begin());
188 #ifdef ONE_ACTION_GROUP
189 #else
190                         if(action_group_)get_ui_manager()->remove_action_group(action_group_);
191                         action_group_=Gtk::ActionGroup::create();
192 #endif
193                 }
194         }
195
196         while(!update_connection_list.empty())
197         {
198                 update_connection_list.front().disconnect();
199                 update_connection_list.pop_front();
200         }
201 }
202
203 void
204 LayerActionManager::queue_refresh()
205 {
206         if(queued)
207                 return;
208
209         //queue_refresh_connection.disconnect();
210         queue_refresh_connection=Glib::signal_idle().connect(
211                 sigc::bind_return(
212                         sigc::mem_fun(*this,&LayerActionManager::refresh),
213                         false
214                 )
215         );
216
217         queued=true;
218 }
219
220 void
221 LayerActionManager::refresh()
222 {
223         if(queued)
224         {
225                 queued=false;
226                 //queue_refresh_connection.disconnect();
227         }
228
229
230         clear();
231
232         // Make sure we are ready
233         if(!ui_manager_ || !layer_tree_ || !canvas_interface_)
234         {
235                 synfig::error("LayerActionManager::refresh(): Not ready!");
236                 return;
237         }
238
239         String ui_info;
240
241         action_paste_->set_sensitive(!clipboard_.empty());
242         action_group_->add(action_paste_);
243
244         if(layer_tree_->get_selection()->count_selected_rows()!=0)
245         {
246                 bool multiple_selected(layer_tree_->get_selection()->count_selected_rows()>1);
247                 Layer::Handle layer(layer_tree_->get_selected_layer());
248
249                 {
250                         bool canvas_set(false);
251                         synfigapp::Action::ParamList param_list;
252                         param_list.add("time",get_canvas_interface()->get_time());
253                         param_list.add("canvas_interface",get_canvas_interface());
254                         {
255                                 synfigapp::SelectionManager::LayerList layer_list(layer_tree_->get_selected_layers());
256                                 synfigapp::SelectionManager::LayerList::iterator iter;
257                                 action_copy_->set_sensitive(!layer_list.empty());
258                                 action_cut_->set_sensitive(!layer_list.empty());
259                                 action_group_->add(action_copy_);
260                                 action_group_->add(action_cut_);
261
262                                 action_amount_inc_->set_sensitive(!layer_list.empty());
263                                 action_amount_dec_->set_sensitive(!layer_list.empty());
264                                 action_amount_->set_sensitive(!layer_list.empty());
265                                 action_group_->add(action_amount_inc_);
266                                 action_group_->add(action_amount_dec_);
267                                 action_group_->add(action_amount_);
268
269                                 for(iter=layer_list.begin();iter!=layer_list.end();++iter)
270                                 {
271                                         update_connection_list.push_back(
272                                                 (*iter)->signal_changed().connect(
273                                                         sigc::mem_fun(*this, &LayerActionManager::queue_refresh)
274                                                 )
275                                         );
276
277                                         if(!canvas_set)
278                                         {
279                                                 param_list.add("canvas",Canvas::Handle((*iter)->get_canvas()));
280                                                 canvas_set=true;
281                                                 update_connection_list.push_back(
282                                                         (*iter)->get_canvas()->signal_changed().connect(
283                                                                 sigc::mem_fun(*this, &LayerActionManager::queue_refresh)
284                                                         )
285                                                 );
286                                         }
287                                         param_list.add("layer",Layer::Handle(*iter));
288                                 }
289                         }
290
291                         if(!multiple_selected && layer->get_name()=="PasteCanvas")
292                         {
293                                 if (select_all_child_layers_connection)
294                                         select_all_child_layers_connection.disconnect();
295
296                                 select_all_child_layers_connection = action_select_all_child_layers_->signal_activate().connect(
297                                         sigc::bind(sigc::mem_fun(*layer_tree_,
298                                                                                          &studio::LayerTree::select_all_children_layers),
299                                                            Layer::LooseHandle(layer)));
300
301                                 action_select_all_child_layers_->set_sensitive(true);
302
303                                 ui_info+="<menuitem action='select-all-child-layers'/>";
304                         }
305                         else
306                                 action_select_all_child_layers_->set_sensitive(false);
307                         action_group_->add(action_select_all_child_layers_);
308
309                         handle<studio::Instance>::cast_static(get_canvas_interface()->get_instance())->
310                                 add_actions_to_group(action_group_, ui_info,   param_list, synfigapp::Action::CATEGORY_LAYER);
311                 }
312         }
313
314         ui_info=("<ui>"
315                            "<popup action='menu-main'>"
316                              "<menu action='menu-layer'>" +
317                                    ui_info +
318                                    "<separator/>"
319                                "<menuitem action='cut' />"
320                                    "<menuitem action='copy' />"
321                                    "<menuitem action='paste' />"
322                                    "<separator/>"
323                              "</menu>"
324                            "</popup>" +
325                          "</ui>");
326         popup_id_=get_ui_manager()->add_ui_from_string(ui_info);
327 #ifdef ONE_ACTION_GROUP
328 #else
329         get_ui_manager()->insert_action_group(action_group_);
330 #endif
331 }
332
333 void
334 LayerActionManager::cut()
335 {
336         copy();
337         if(action_group_->get_action("action-layer_remove"))
338                 action_group_->get_action("action-layer_remove")->activate();
339 }
340
341 void
342 LayerActionManager::copy()
343 {
344         synfigapp::SelectionManager::LayerList layer_list(layer_tree_->get_selected_layers());
345         clipboard_.clear();
346         synfig::GUID guid;
347
348         while(!layer_list.empty())
349         {
350                 clipboard_.push_back(layer_list.front()->clone(guid));
351                 layer_list.pop_front();
352         }
353
354         action_paste_->set_sensitive(!clipboard_.empty());
355
356         //queue_refresh();
357 }
358
359 void
360 LayerActionManager::paste()
361 {
362         synfig::GUID guid;
363
364         // Create the action group
365         synfigapp::Action::PassiveGrouper group(get_canvas_interface()->get_instance().get(),_("Paste"));
366
367         Canvas::Handle canvas(get_canvas_interface()->get_canvas());
368         int depth(0);
369
370         // we are temporarily using the layer to hold something
371         Layer::Handle layer(layer_tree_->get_selected_layer());
372         if(layer)
373         {
374                 depth=layer->get_depth();
375                 canvas=layer->get_canvas();
376         }
377
378         synfigapp::SelectionManager::LayerList layer_selection;
379
380         for(std::list<synfig::Layer::Handle>::iterator iter=clipboard_.begin();iter!=clipboard_.end();++iter)
381         {
382                 layer=(*iter)->clone(guid);
383                 layer_selection.push_back(layer);
384                 synfigapp::Action::Handle       action(synfigapp::Action::create("layer_add"));
385
386                 assert(action);
387                 if(!action)
388                         return;
389
390                 action->set_param("canvas",canvas);
391                 action->set_param("canvas_interface",etl::loose_handle<synfigapp::CanvasInterface>(get_canvas_interface()));
392                 action->set_param("new",layer);
393
394                 if(!action->is_ready())
395                 {
396                         return;
397                 }
398
399                 if(!get_instance()->perform_action(action))
400                 {
401                         return;
402                 }
403
404                 // synfig::info("DEPTH=%d",depth);
405
406                 // Action to move the layer (if necessary)
407                 if(depth>0)
408                 {
409                         synfigapp::Action::Handle       action(synfigapp::Action::create("layer_move"));
410
411                         assert(action);
412                         if(!action)
413                                 return;
414
415                         action->set_param("canvas",canvas);
416                         action->set_param("canvas_interface",etl::loose_handle<synfigapp::CanvasInterface>(get_canvas_interface()));
417                         action->set_param("layer",layer);
418                         action->set_param("new_index",depth);
419
420                         if(!action->is_ready())
421                         {
422                                 //get_ui_interface()->error(_("Move Action Not Ready"));
423                                 //return 0;
424                                 return;
425                         }
426
427                         if(!get_instance()->perform_action(action))
428                         {
429                                 //get_ui_interface()->error(_("Move Action Not Ready"));
430                                 //return 0;
431                                 return;
432                         }
433                 }
434                 depth++;
435
436                 // automatically export the Index parameter of Duplicate layers when pasting
437                 int index = 1;
438                 export_dup_nodes(layer, canvas, index);
439         }
440         get_canvas_interface()->get_selection_manager()->clear_selected_layers();
441         get_canvas_interface()->get_selection_manager()->set_selected_layers(layer_selection);
442 }
443
444 void
445 LayerActionManager::export_dup_nodes(synfig::Layer::Handle layer, Canvas::Handle canvas, int &index)
446 {
447         // automatically export the Index parameter of Duplicate layers when pasting
448         if (layer->get_name() == "duplicate")
449                 while (true)
450                 {
451                         String name = strprintf(_("Index %d"), index++);
452                         try
453                         {
454                                 canvas->find_value_node(name);
455                         }
456                         catch (Exception::IDNotFound x)
457                         {
458                                 get_canvas_interface()->add_value_node(layer->dynamic_param_list().find("index")->second, name);
459                                 break;
460                         }
461                 }
462         else
463         {
464                 Layer::ParamList param_list(layer->get_param_list());
465                 for (Layer::ParamList::const_iterator iter(param_list.begin())
466                                  ; iter != param_list.end()
467                                  ; iter++)
468                         if (layer->dynamic_param_list().count(iter->first)==0 && iter->second.get_type()==ValueBase::TYPE_CANVAS)
469                         {
470                                 Canvas::Handle subcanvas(iter->second.get(Canvas::Handle()));
471                                 if (subcanvas && subcanvas->is_inline())
472                                         for (Context iter = subcanvas->get_context(); iter != subcanvas->end(); iter++)
473                                                 export_dup_nodes(*iter, canvas, index);
474                         }
475
476                 for (Layer::DynamicParamList::const_iterator iter(layer->dynamic_param_list().begin())
477                                  ; iter != layer->dynamic_param_list().end()
478                                  ; iter++)
479                         if (iter->second->get_type()==ValueBase::TYPE_CANVAS)
480                         {
481                                 Canvas::Handle canvas((*iter->second)(0).get(Canvas::Handle()));
482                                 if (canvas->is_inline())
483                                         //! \todo do we need to implement this?  and if so, shouldn't we check all canvases, not just the one at t=0s?
484                                         warning("%s:%d not yet implemented - do we need to export duplicate valuenodes in dynamic canvas parameters?", __FILE__, __LINE__);
485                         }
486         }
487 }
488
489 void
490 LayerActionManager::amount_inc()
491 {
492         float adjust(0.1);
493
494         // Create the action group
495         synfigapp::Action::PassiveGrouper group(get_canvas_interface()->get_instance().get(),_("Decrease Amount"));
496
497         if(adjust>0)
498                 group.set_name(_("Increase Amount"));
499
500         synfigapp::SelectionManager::LayerList layer_list(layer_tree_->get_selected_layers());
501
502         while(!layer_list.empty())
503         {
504                 ValueBase value(layer_list.front()->get_param("amount"));
505                 if(value.same_type_as(Real()))
506                 {
507                         get_canvas_interface()->change_value(synfigapp::ValueDesc(layer_list.front(),"amount"),value.get(Real())+adjust);
508                 }
509                 layer_list.pop_front();
510         }
511 }
512
513 void
514 LayerActionManager::amount_dec()
515 {
516         float adjust(-0.1);
517
518         // Create the action group
519         synfigapp::Action::PassiveGrouper group(get_canvas_interface()->get_instance().get(),_("Decrease Amount"));
520
521         if(adjust>0)
522                 group.set_name(_("Increase Amount"));
523
524         synfigapp::SelectionManager::LayerList layer_list(layer_tree_->get_selected_layers());
525
526         while(!layer_list.empty())
527         {
528                 ValueBase value(layer_list.front()->get_param("amount"));
529                 if(value.same_type_as(Real()))
530                 {
531                         get_canvas_interface()->change_value(synfigapp::ValueDesc(layer_list.front(),"amount"),value.get(Real())+adjust);
532                 }
533                 layer_list.pop_front();
534         }
535 }