6a752d2b9b88a85e0e05656139f003b4d5fe4744
[synfig.git] / synfig-studio / src / gui / states / state_text.cpp
1 /* === S Y N F I G ========================================================= */
2 /*!     \file state_text.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) 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 <gtkmm/dialog.h>
34 #include <gtkmm/entry.h>
35
36 #include "state_text.h"
37 #include "state_normal.h"
38 #include "canvasview.h"
39 #include "workarea.h"
40 #include "app.h"
41
42 #include <synfigapp/action.h>
43 #include "event_mouse.h"
44 #include "event_layerclick.h"
45 #include "toolbox.h"
46 #include "docks/dialog_tooloptions.h"
47 #include <gtkmm/optionmenu.h>
48 #include "duck.h"
49 #include "widgets/widget_enum.h"
50 #include <synfigapp/main.h>
51
52 #include "general.h"
53
54 #endif
55
56 /* === U S I N G =========================================================== */
57
58 using namespace std;
59 using namespace etl;
60 using namespace synfig;
61 using namespace studio;
62
63 /* === M A C R O S ========================================================= */
64
65 /* === G L O B A L S ======================================================= */
66
67 StateText studio::state_text;
68
69 /* === C L A S S E S & S T R U C T S ======================================= */
70
71 class studio::StateText_Context
72 {
73         etl::handle<CanvasView> canvas_view;
74         CanvasView::IsWorking is_working;
75
76         Duckmatic::Push duckmatic_push;
77
78         void refresh_ducks();
79
80         bool prev_workarea_layer_status_;
81
82         //Toolbox settings
83         synfigapp::Settings& settings;
84
85         //Toolbox display
86         Gtk::Table options_table;
87
88         Gtk::Entry              entry_id; //what to name the layer
89         Gtk::Entry              entry_family;
90         Widget_Vector   widget_size;
91         Widget_Vector   widget_orientation;
92         Gtk::CheckButton checkbutton_paragraph;
93
94 public:
95         synfig::String get_id()const { return entry_id.get_text(); }
96         void set_id(const synfig::String& x) { return entry_id.set_text(x); }
97
98         bool get_paragraph_flag()const { return checkbutton_paragraph.get_active(); }
99         void set_paragraph_flag(bool x) { return checkbutton_paragraph.set_active(x); }
100
101         Vector get_size() { return widget_size.get_value(); }
102         void set_size(Vector s) { return widget_size.set_value(s); }
103
104         Vector get_orientation() { return widget_orientation.get_value(); }
105         void set_orientation(Vector s) { return widget_orientation.set_value(s); }
106
107         String get_family()const { return entry_family.get_text(); }
108         void set_family(String s) { return entry_family.set_text(s); }
109
110         void refresh_tool_options(); //to refresh the toolbox
111
112         //events
113         Smach::event_result event_stop_handler(const Smach::event& x);
114         Smach::event_result event_refresh_handler(const Smach::event& x);
115         Smach::event_result event_mouse_click_handler(const Smach::event& x);
116         Smach::event_result event_refresh_tool_options(const Smach::event& x);
117         Smach::event_result event_workarea_mouse_button_down_handler(const Smach::event& x);
118
119         //constructor destructor
120         StateText_Context(CanvasView *canvas_view);
121         ~StateText_Context();
122
123         const etl::handle<CanvasView>& get_canvas_view()const{return canvas_view;}
124         etl::handle<synfigapp::CanvasInterface> get_canvas_interface()const{return canvas_view->canvas_interface();}
125         WorkArea * get_work_area()const{return canvas_view->get_work_area();}
126
127         //Modifying settings etc.
128         void load_settings();
129         void save_settings();
130         void reset();
131         void increment_id();
132         bool egress_on_selection_change;
133         Smach::event_result event_layer_selection_changed_handler(const Smach::event& /*x*/)
134         {
135                 if(egress_on_selection_change)
136                         throw &state_normal; //throw Smach::egress_exception();
137                 return Smach::RESULT_OK;
138         }
139
140         void make_text(const Point& point);
141
142 }; // END of class StateText_Context
143
144 /* === P R O C E D U R E S ================================================= */
145
146 /* === M E T H O D S ======================================================= */
147
148 StateText::StateText():
149         Smach::state<StateText_Context>("text")
150 {
151         insert(event_def(EVENT_LAYER_SELECTION_CHANGED,&StateText_Context::event_layer_selection_changed_handler));
152         insert(event_def(EVENT_STOP,&StateText_Context::event_stop_handler));
153         insert(event_def(EVENT_REFRESH,&StateText_Context::event_refresh_handler));
154         insert(event_def(EVENT_REFRESH_DUCKS,&StateText_Context::event_refresh_handler));
155         insert(event_def(EVENT_WORKAREA_MOUSE_BUTTON_DOWN,&StateText_Context::event_workarea_mouse_button_down_handler));
156         insert(event_def(EVENT_REFRESH_TOOL_OPTIONS,&StateText_Context::event_refresh_tool_options));
157 }
158
159 StateText::~StateText()
160 {
161 }
162
163 void
164 StateText_Context::load_settings()
165 {
166         try
167         {
168                 SETTINGS_LOCALE_SAFE_AND_BACKUP
169                 String value;
170                 Vector v;
171
172                 //parse the arguments yargh!
173                 if(settings.get_value("text.id",value))
174                         set_id(value);
175                 else
176                         set_id("Text");
177
178                 if(settings.get_value("text.paragraph",value) && value=="1")
179                         set_paragraph_flag(true);
180                 else
181                         set_paragraph_flag(false);
182
183                 if(settings.get_value("text.size_x",value))
184                         v[0] = atof(value.c_str());
185                 else
186                         v[0] = 0.25;
187                 if(settings.get_value("text.size_y",value))
188                         v[1] = atof(value.c_str());
189                 else
190                         v[1] = 0.25;
191                 set_size(v);
192
193                 if(settings.get_value("text.orient_x",value))
194                         v[0] = atof(value.c_str());
195                 else
196                         v[0] = 0.5;
197                 if(settings.get_value("text.orient_y",value))
198                         v[1] = atof(value.c_str());
199                 else
200                         v[1] = 0.5;
201                 set_orientation(v);
202
203                 if(settings.get_value("text.family",value))
204                         set_family(value);
205                 else
206                         set_family("Sans Serif");
207                 SETTINGS_LOCALE_RESTORE
208         }
209         catch(...)
210         {
211                 synfig::warning("State Text: Caught exception when attempting to load settings.");
212         }
213 }
214
215 void
216 StateText_Context::save_settings()
217 {
218         try
219         {
220                 SETTINGS_LOCALE_SAFE_AND_BACKUP
221                 settings.set_value("text.id",get_id());
222                 settings.set_value("text.paragraph",get_paragraph_flag()?"1":"0");
223                 settings.set_value("text.size_x",strprintf("%f",(float)get_size()[0]));
224                 settings.set_value("text.size_y",strprintf("%f",(float)get_size()[1]));
225                 settings.set_value("text.orient_x",strprintf("%f",(float)get_orientation()[0]));
226                 settings.set_value("text.orient_y",strprintf("%f",(float)get_orientation()[1]));
227                 settings.set_value("text.family",get_family());
228                 SETTINGS_LOCALE_RESTORE
229         }
230         catch(...)
231         {
232                 synfig::warning("State Text: Caught exception when attempting to save settings.");
233         }
234 }
235
236 void
237 StateText_Context::reset()
238 {
239         refresh_ducks();
240 }
241
242 void
243 StateText_Context::increment_id()
244 {
245         String id(get_id());
246         int number=1;
247         int digits=0;
248
249         if(id.empty())
250                 id="Text";
251
252         // If there is a number
253         // already at the end of the
254         // id, then remove it.
255         if(id[id.size()-1]<='9' && id[id.size()-1]>='0')
256         {
257                 // figure out how many digits it is
258                 for (digits = 0;
259                          (int)id.size()-1 >= digits && id[id.size()-1-digits] <= '9' && id[id.size()-1-digits] >= '0';
260                          digits++)
261                         ;
262
263                 String str_number;
264                 str_number=String(id,id.size()-digits,id.size());
265                 id=String(id,0,id.size()-digits);
266
267                 number=atoi(str_number.c_str());
268         }
269         else
270         {
271                 number=1;
272                 digits=3;
273         }
274
275         number++;
276
277         // Add the number back onto the id
278         {
279                 const String format(strprintf("%%0%dd",digits));
280                 id+=strprintf(format.c_str(),number);
281         }
282
283         // Set the ID
284         set_id(id);
285 }
286
287 StateText_Context::StateText_Context(CanvasView *canvas_view):
288         canvas_view(canvas_view),
289         is_working(*canvas_view),
290         duckmatic_push(get_work_area()),
291         prev_workarea_layer_status_(get_work_area()->get_allow_layer_clicks()),
292         settings(synfigapp::Main::get_selected_input_device()->settings()),
293         entry_id(),
294         checkbutton_paragraph(_("Multiline Editor"))
295 {
296         egress_on_selection_change=true;
297
298         widget_size.set_digits(2);
299         widget_size.set_canvas(canvas_view->get_canvas());
300
301         widget_orientation.set_digits(2);
302
303         options_table.attach(*manage(new Gtk::Label(_("Text Tool"))),           0, 2, 0, 1, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
304         options_table.attach(entry_id,                                                                          0, 2, 1, 2, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
305         options_table.attach(checkbutton_paragraph,                                                     0, 2, 2, 3, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
306         options_table.attach(*manage(new Gtk::Label(_("Size:"))),                       0, 1, 3, 4, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
307         options_table.attach(widget_size,                                                                       1, 2, 3, 4, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
308         options_table.attach(*manage(new Gtk::Label(_("Orientation:"))),        0, 1, 4, 5, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
309         options_table.attach(widget_orientation,                                                        1, 2, 4, 5, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
310         options_table.attach(*manage(new Gtk::Label(_("Family:"))),                     0, 1, 5, 6, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
311         options_table.attach(entry_family,                                                                      1, 2, 5, 6, Gtk::EXPAND|Gtk::FILL, Gtk::EXPAND|Gtk::FILL, 0, 0);
312
313         load_settings();
314
315         options_table.show_all();
316
317         refresh_tool_options();
318         App::dialog_tool_options->present();
319
320         // Turn off layer clicking
321         get_work_area()->set_allow_layer_clicks(false);
322
323         // clear out the ducks
324         get_work_area()->clear_ducks();
325
326         // Refresh the work area
327         get_work_area()->queue_draw();
328
329         // Hide the tables if they are showing
330         //prev_table_status=get_canvas_view()->tables_are_visible();
331         //if(prev_table_status)get_canvas_view()->hide_tables();
332
333         // Disable the time bar
334         //get_canvas_view()->set_sensitive_timebar(false);
335
336         // Connect a signal
337         //get_work_area()->signal_user_click().connect(sigc::mem_fun(*this,&studio::StateText_Context::on_user_click));
338         get_work_area()->set_cursor(Gdk::XTERM);
339
340         App::toolbox->refresh();
341 }
342
343 void
344 StateText_Context::refresh_tool_options()
345 {
346         App::dialog_tool_options->clear();
347         App::dialog_tool_options->set_widget(options_table);
348         App::dialog_tool_options->set_local_name(_("Text Tool"));
349         App::dialog_tool_options->set_name("text");
350 }
351
352 Smach::event_result
353 StateText_Context::event_refresh_tool_options(const Smach::event& /*x*/)
354 {
355         refresh_tool_options();
356         return Smach::RESULT_ACCEPT;
357 }
358
359 StateText_Context::~StateText_Context()
360 {
361         save_settings();
362
363         // Restore layer clicking
364         get_work_area()->set_allow_layer_clicks(prev_workarea_layer_status_);
365         get_work_area()->reset_cursor();
366
367         App::dialog_tool_options->clear();
368
369         get_work_area()->queue_draw();
370
371         get_canvas_view()->queue_rebuild_ducks();
372
373         App::toolbox->refresh();
374 }
375
376 Smach::event_result
377 StateText_Context::event_stop_handler(const Smach::event& /*x*/)
378 {
379         //throw Smach::egress_exception();
380         throw &state_normal;
381         return Smach::RESULT_OK;
382 }
383
384 Smach::event_result
385 StateText_Context::event_refresh_handler(const Smach::event& /*x*/)
386 {
387         refresh_ducks();
388         return Smach::RESULT_ACCEPT;
389 }
390
391 void
392 StateText_Context::make_text(const Point& _point)
393 {
394         synfigapp::Action::PassiveGrouper group(get_canvas_interface()->get_instance().get(),_("New Text"));
395         synfigapp::PushMode push_mode(get_canvas_interface(),synfigapp::MODE_NORMAL);
396
397         Layer::Handle layer;
398
399         Canvas::Handle canvas(get_canvas_view()->get_canvas());
400         int depth(0);
401
402         // we are temporarily using the layer to hold something
403         layer=get_canvas_view()->get_selection_manager()->get_selected_layer();
404         if(layer)
405         {
406                 depth=layer->get_depth();
407                 canvas=layer->get_canvas();
408         }
409
410         synfigapp::SelectionManager::LayerList layer_selection;
411         if (!getenv("SYNFIG_TOOLS_CLEAR_SELECTION"))
412                 layer_selection = get_canvas_view()->get_selection_manager()->get_selected_layers();
413
414         const synfig::TransformStack& transform(get_canvas_view()->get_curr_transform_stack());
415         const Point point(transform.unperform(_point));
416
417         String text;
418         if (get_paragraph_flag())
419                 App::dialog_paragraph(_("Text Paragraph"), _("Enter text here:"), text);
420         else
421                 App::dialog_entry(_("Text Entry"), _("Enter text here:"), text);
422
423         layer=get_canvas_interface()->add_layer_to("text",canvas,depth);
424         if (!layer)
425         {
426                 get_canvas_view()->get_ui_interface()->error(_("Unable to create layer"));
427                 group.cancel();
428                 return;
429         }
430         layer_selection.push_back(layer);
431
432         layer->set_param("origin",point);
433         get_canvas_interface()->signal_layer_param_changed()(layer,"origin");
434
435         layer->set_param("text",text);
436         get_canvas_interface()->signal_layer_param_changed()(layer,"text");
437
438         layer->set_param("size",get_size());
439         get_canvas_interface()->signal_layer_param_changed()(layer,"size");
440
441         layer->set_param("orient",get_orientation());
442         get_canvas_interface()->signal_layer_param_changed()(layer,"orient");
443
444         layer->set_param("family",get_family());
445         get_canvas_interface()->signal_layer_param_changed()(layer,"family");
446
447         layer->set_description(get_id());
448         get_canvas_interface()->signal_layer_new_description()(layer,layer->get_description());
449
450         egress_on_selection_change=false;
451         get_canvas_interface()->get_selection_manager()->clear_selected_layers();
452         get_canvas_interface()->get_selection_manager()->set_selected_layers(layer_selection);
453         egress_on_selection_change=true;
454
455         reset();
456         increment_id();
457 }
458
459 Smach::event_result
460 StateText_Context::event_workarea_mouse_button_down_handler(const Smach::event& x)
461 {
462         const EventMouse& event(*reinterpret_cast<const EventMouse*>(&x));
463         if(event.button==BUTTON_LEFT)
464         {
465                 make_text(get_work_area()->snap_point_to_grid(event.pos));
466
467                 get_work_area()->clear_ducks();
468                 return Smach::RESULT_ACCEPT;
469         }
470         return Smach::RESULT_OK;
471 }
472
473 void
474 StateText_Context::refresh_ducks()
475 {
476         get_work_area()->clear_ducks();
477         get_work_area()->queue_draw();
478 }