1 /*
2  * Sylpheed templates subsystem
3  * Copyright (C) 2001 Alexander Barinov
4  * Copyright (C) 2001-2015 Hiroyuki Yamamoto
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 2 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program; if not, write to the Free Software
18  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
19  */
20 
21 #include "defs.h"
22 
23 #include <glib.h>
24 #include <glib/gi18n.h>
25 #include <gtk/gtk.h>
26 #include <gdk/gdkkeysyms.h>
27 #include <string.h>
28 #include <dirent.h>
29 #include <sys/stat.h>
30 
31 #include "template.h"
32 #include "main.h"
33 #include "inc.h"
34 #include "utils.h"
35 #include "gtkutils.h"
36 #include "alertpanel.h"
37 #include "manage_window.h"
38 #include "prefs_common.h"
39 #include "prefs_common_dialog.h"
40 #include "compose.h"
41 #include "mainwindow.h"
42 #include "addr_compl.h"
43 #include "quote_fmt.h"
44 #include "plugin.h"
45 
46 static struct Templates {
47 	GtkWidget *window;
48 	GtkWidget *clist_tmpls;
49 	GtkWidget *entry_name;
50 	GtkWidget *entry_to;
51 	GtkWidget *entry_cc;
52 	GtkWidget *entry_bcc;
53 	GtkWidget *entry_replyto;
54 	GtkWidget *entry_subject;
55 	GtkWidget *text_value;
56 	GtkWidget *confirm_area;
57 	GtkWidget *ok_btn;
58 	GtkWidget *cancel_btn;
59 
60 	gboolean entry_modified;
61 	gboolean list_modified;
62 } templates;
63 
64 /* widget creating functions */
65 static void prefs_template_window_create	(void);
66 static void prefs_template_window_setup		(void);
67 static void prefs_template_clear		(void);
68 
69 static GSList *prefs_template_get_list		(void);
70 
71 /* callbacks */
72 static gint prefs_template_deleted_cb		(GtkWidget	*widget,
73 						 GdkEventAny	*event,
74 						 gpointer	 data);
75 static gboolean prefs_template_key_pressed_cb	(GtkWidget	*widget,
76 						 GdkEventKey	*event,
77 						 gpointer	 data);
78 static void prefs_template_changed_cb		(GtkEditable	*editable,
79 						 gpointer	 data);
80 static void prefs_template_cancel_cb		(void);
81 static void prefs_template_ok_cb		(void);
82 static void prefs_template_select_cb		(GtkCList	*clist,
83 						 gint		 row,
84 						 gint		 column,
85 						 GdkEvent	*event);
86 static void prefs_template_register_cb		(void);
87 static void prefs_template_substitute_cb	(void);
88 static void prefs_template_delete_cb		(void);
89 static void prefs_template_up_cb		(void);
90 static void prefs_template_down_cb		(void);
91 
92 /* Called from mainwindow.c */
prefs_template_open(void)93 void prefs_template_open(void)
94 {
95 	inc_lock();
96 
97 	if (!templates.window)
98 		prefs_template_window_create();
99 
100 	prefs_template_window_setup();
101 	gtk_widget_show(templates.window);
102 
103 	syl_plugin_signal_emit("prefs-template-open", templates.window);
104 }
105 
106 #define ADD_ENTRY(entry, str, row) \
107 { \
108 	label1 = gtk_label_new(str); \
109 	gtk_widget_show(label1); \
110 	gtk_table_attach(GTK_TABLE(table), label1, 0, 1, row, (row + 1), \
111 			 GTK_FILL, 0, 0, 0); \
112 	gtk_misc_set_alignment(GTK_MISC(label1), 1, 0.5); \
113  \
114 	entry = gtk_entry_new(); \
115 	gtk_widget_show(entry); \
116 	gtk_table_attach(GTK_TABLE(table), entry, 1, 2, row, (row + 1), \
117 			 GTK_EXPAND|GTK_SHRINK|GTK_FILL, 0, 0, 0); \
118 	g_signal_connect(G_OBJECT(entry), "changed", \
119 			 G_CALLBACK(prefs_template_changed_cb), NULL); \
120 }
121 
prefs_template_window_create(void)122 static void prefs_template_window_create(void)
123 {
124 	/* window structure ;) */
125 	GtkWidget *window;
126 	GtkWidget   *vpaned;
127 	GtkWidget     *vbox1;
128 	GtkWidget       *hbox1;
129 	GtkWidget         *label1;
130 	GtkWidget         *entry_name;
131 	GtkWidget       *table;
132 	GtkWidget         *entry_to;
133 	GtkWidget         *entry_cc;
134 	GtkWidget         *entry_bcc;
135 	GtkWidget         *entry_replyto;
136 	GtkWidget         *entry_subject;
137 	GtkWidget       *scroll2;
138 	GtkWidget         *text_value;
139 	GtkWidget     *vbox2;
140 	GtkWidget       *hbox2;
141 	GtkWidget         *arrow1;
142 	GtkWidget         *hbox3;
143 	GtkWidget           *reg_btn;
144 	GtkWidget           *subst_btn;
145 	GtkWidget           *del_btn;
146 	GtkWidget         *desc_btn;
147 	GtkWidget       *hbox4;
148 	GtkWidget         *scroll1;
149 	GtkWidget           *clist_tmpls;
150 	GtkWidget         *vbox3;
151 	GtkWidget           *vbox4;
152 	GtkWidget             *up_btn;
153 	GtkWidget             *down_btn;
154 	GtkWidget       *confirm_area;
155 	GtkWidget         *ok_btn;
156 	GtkWidget         *cancel_btn;
157 
158 	GtkTextBuffer *buffer;
159 
160 	gchar *title[1];
161 
162 	/* main window */
163 	window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
164 	gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
165 	gtk_window_set_modal(GTK_WINDOW(window), TRUE);
166 	gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, FALSE);
167 
168 	/* vpaned to separate template settings from templates list */
169 	vpaned = gtk_vpaned_new();
170 	gtk_widget_show(vpaned);
171 	gtk_container_add(GTK_CONTAINER(window), vpaned);
172 
173 	/* vbox to handle template name and content */
174 	vbox1 = gtk_vbox_new(FALSE, 6);
175 	gtk_widget_show(vbox1);
176 	gtk_container_set_border_width(GTK_CONTAINER(vbox1), 8);
177 	gtk_paned_pack1(GTK_PANED(vpaned), vbox1, FALSE, FALSE);
178 
179 	hbox1 = gtk_hbox_new(FALSE, 8);
180 	gtk_widget_show(hbox1);
181 	gtk_box_pack_start(GTK_BOX(vbox1), hbox1, FALSE, FALSE, 0);
182 
183 	label1 = gtk_label_new(_("Template name"));
184 	gtk_widget_show(label1);
185 	gtk_box_pack_start(GTK_BOX(hbox1), label1, FALSE, FALSE, 0);
186 
187 	entry_name = gtk_entry_new();
188 	gtk_widget_show(entry_name);
189 	gtk_box_pack_start(GTK_BOX(hbox1), entry_name, TRUE, TRUE, 0);
190 	g_signal_connect(G_OBJECT(entry_name), "changed",
191 			 G_CALLBACK(prefs_template_changed_cb), NULL);
192 
193 	/* table for headers */
194 	table = gtk_table_new(5, 2, FALSE);
195 	gtk_widget_show(table);
196 	gtk_box_pack_start(GTK_BOX(vbox1), table, FALSE, FALSE, 0);
197 	gtk_table_set_row_spacings(GTK_TABLE(table), 4);
198 	gtk_table_set_col_spacings(GTK_TABLE(table), 4);
199 
200 	ADD_ENTRY(entry_to, _("To:"), 0);
201 	address_completion_register_entry(GTK_ENTRY(entry_to));
202 	ADD_ENTRY(entry_cc, _("Cc:"), 1);
203 	address_completion_register_entry(GTK_ENTRY(entry_cc));
204 	ADD_ENTRY(entry_bcc, _("Bcc:"), 2);
205 	address_completion_register_entry(GTK_ENTRY(entry_bcc));
206 	ADD_ENTRY(entry_replyto, _("Reply-To:"), 3);
207 	address_completion_register_entry(GTK_ENTRY(entry_replyto));
208 	ADD_ENTRY(entry_subject, _("Subject:"), 4);
209 
210 #undef ADD_ENTRY
211 
212 	/* template content */
213 	scroll2 = gtk_scrolled_window_new(NULL, NULL);
214 	gtk_widget_show(scroll2);
215 	gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll2),
216 				       GTK_POLICY_NEVER,
217 				       GTK_POLICY_AUTOMATIC);
218 	gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scroll2),
219 					    GTK_SHADOW_IN);
220 	gtk_box_pack_start(GTK_BOX(vbox1), scroll2, TRUE, TRUE, 0);
221 
222 	text_value = gtk_text_view_new();
223 	gtk_widget_show(text_value);
224 	gtk_widget_set_size_request
225 		(text_value,
226 		 360 * gtkut_get_dpi_multiplier(),
227 		 120 * gtkut_get_dpi_multiplier());
228 	gtk_container_add(GTK_CONTAINER(scroll2), text_value);
229 	gtk_text_view_set_editable(GTK_TEXT_VIEW(text_value), TRUE);
230 	buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(text_value));
231 	g_signal_connect(G_OBJECT(buffer), "changed",
232 			 G_CALLBACK(prefs_template_changed_cb), NULL);
233 
234 	/* vbox for buttons and templates list */
235 	vbox2 = gtk_vbox_new(FALSE, 6);
236 	gtk_widget_show(vbox2);
237 	gtk_container_set_border_width(GTK_CONTAINER(vbox2), 8);
238 	gtk_paned_pack2(GTK_PANED(vpaned), vbox2, TRUE, FALSE);
239 
240 	/* register | substitute | delete */
241 	hbox2 = gtk_hbox_new(FALSE, 4);
242 	gtk_widget_show(hbox2);
243 	gtk_box_pack_start(GTK_BOX(vbox2), hbox2, FALSE, FALSE, 0);
244 
245 	arrow1 = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
246 	gtk_widget_show(arrow1);
247 	gtk_box_pack_start(GTK_BOX(hbox2), arrow1, FALSE, FALSE, 0);
248 	gtk_widget_set_size_request(arrow1, -1,
249 				    16 * gtkut_get_dpi_multiplier());
250 
251 	hbox3 = gtk_hbox_new(TRUE, 4);
252 	gtk_widget_show(hbox3);
253 	gtk_box_pack_start(GTK_BOX(hbox2), hbox3, FALSE, FALSE, 0);
254 
255 	reg_btn = gtk_button_new_with_label(_("Register"));
256 	gtk_widget_show(reg_btn);
257 	gtk_box_pack_start(GTK_BOX(hbox3), reg_btn, FALSE, TRUE, 0);
258 	g_signal_connect(G_OBJECT (reg_btn), "clicked",
259 			 G_CALLBACK (prefs_template_register_cb), NULL);
260 
261 	subst_btn = gtk_button_new_with_label(_(" Substitute "));
262 	gtk_widget_show(subst_btn);
263 	gtk_box_pack_start(GTK_BOX(hbox3), subst_btn, FALSE, TRUE, 0);
264 	g_signal_connect(G_OBJECT(subst_btn), "clicked",
265 			 G_CALLBACK(prefs_template_substitute_cb), NULL);
266 
267 	del_btn = gtk_button_new_with_label(_("Delete"));
268 	gtk_widget_show(del_btn);
269 	gtk_box_pack_start(GTK_BOX(hbox3), del_btn, FALSE, TRUE, 0);
270 	g_signal_connect(G_OBJECT(del_btn), "clicked",
271 			 G_CALLBACK(prefs_template_delete_cb), NULL);
272 
273 	desc_btn = gtk_button_new_with_label(_(" Symbols "));
274 	gtk_widget_show(desc_btn);
275 	gtk_box_pack_end(GTK_BOX(hbox2), desc_btn, FALSE, FALSE, 0);
276 	g_signal_connect(G_OBJECT(desc_btn), "clicked",
277 			 G_CALLBACK(prefs_quote_description), NULL);
278 
279 	/* templates list */
280 	hbox4 = gtk_hbox_new(FALSE, 8);
281 	gtk_widget_show(hbox4);
282 	gtk_box_pack_start(GTK_BOX(vbox2), hbox4, TRUE, TRUE, 0);
283 
284 	scroll1 = gtk_scrolled_window_new(NULL, NULL);
285 	gtk_widget_show(scroll1);
286 	gtk_box_pack_start(GTK_BOX(hbox4), scroll1, TRUE, TRUE, 0);
287 	gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scroll1),
288 				       GTK_POLICY_AUTOMATIC,
289 				       GTK_POLICY_AUTOMATIC);
290 
291 	title[0] = _("Registered templates");
292 	clist_tmpls = gtk_clist_new_with_titles(1, title);
293 	gtk_widget_show(clist_tmpls);
294 	gtk_widget_set_size_request(scroll1, -1,
295 				    160 * gtkut_get_dpi_multiplier());
296 	gtk_container_add(GTK_CONTAINER(scroll1), clist_tmpls);
297 	gtk_clist_set_column_width(GTK_CLIST(clist_tmpls), 0, 80);
298 	gtk_clist_set_selection_mode(GTK_CLIST(clist_tmpls),
299 				     GTK_SELECTION_BROWSE);
300 	gtkut_clist_set_redraw(GTK_CLIST(clist_tmpls));
301 	GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist_tmpls)->column[0].button,
302 			       GTK_CAN_FOCUS);
303 	g_signal_connect(G_OBJECT (clist_tmpls), "select_row",
304 			 G_CALLBACK (prefs_template_select_cb), NULL);
305 
306 	vbox3 = gtk_vbox_new(TRUE, 0);
307 	gtk_widget_show(vbox3);
308 	gtk_box_pack_start(GTK_BOX(hbox4), vbox3, FALSE, FALSE, 0);
309 
310 	vbox4 = gtk_vbox_new(TRUE, 8);
311 	gtk_widget_show(vbox4);
312 	gtk_box_pack_start(GTK_BOX(vbox3), vbox4, TRUE, FALSE, 0);
313 
314 	up_btn = gtk_button_new_with_label(_("Up"));
315 	gtk_widget_show(up_btn);
316 	gtk_box_pack_start(GTK_BOX(vbox4), up_btn, FALSE, FALSE, 0);
317 	g_signal_connect(G_OBJECT (up_btn), "clicked",
318 			 G_CALLBACK (prefs_template_up_cb), NULL);
319 
320 	down_btn = gtk_button_new_with_label(_("Down"));
321 	gtk_widget_show(down_btn);
322 	gtk_box_pack_start(GTK_BOX(vbox4), down_btn, FALSE, FALSE, 0);
323 	g_signal_connect(G_OBJECT (down_btn), "clicked",
324 			 G_CALLBACK (prefs_template_down_cb), NULL);
325 
326 	/* ok | cancel */
327 	gtkut_stock_button_set_create(&confirm_area, &ok_btn, GTK_STOCK_OK,
328 				      &cancel_btn, GTK_STOCK_CANCEL,
329 				      NULL, NULL);
330 	gtk_widget_show(confirm_area);
331 	gtk_box_pack_end(GTK_BOX(vbox2), confirm_area, FALSE, FALSE, 0);
332 	gtk_widget_grab_default(ok_btn);
333 
334 	gtk_window_set_title(GTK_WINDOW(window), _("Templates"));
335 
336 	g_signal_connect(G_OBJECT(window), "delete_event",
337 			 G_CALLBACK(prefs_template_deleted_cb), NULL);
338 	g_signal_connect(G_OBJECT(window), "key_press_event",
339 			 G_CALLBACK(prefs_template_key_pressed_cb), NULL);
340 	MANAGE_WINDOW_SIGNALS_CONNECT(window);
341 	g_signal_connect(G_OBJECT(ok_btn), "clicked",
342 			 G_CALLBACK(prefs_template_ok_cb), NULL);
343 	g_signal_connect(G_OBJECT(cancel_btn), "clicked",
344 			 G_CALLBACK(prefs_template_cancel_cb), NULL);
345 
346 	address_completion_start(window);
347 
348 	templates.window = window;
349 	templates.clist_tmpls = clist_tmpls;
350 	templates.entry_name = entry_name;
351 	templates.entry_to = entry_to;
352 	templates.entry_cc = entry_cc;
353 	templates.entry_bcc = entry_bcc;
354 	templates.entry_replyto = entry_replyto;
355 	templates.entry_subject = entry_subject;
356 	templates.text_value = text_value;
357 	templates.confirm_area = confirm_area;
358 	templates.ok_btn = ok_btn;
359 	templates.cancel_btn = cancel_btn;
360 	templates.entry_modified = FALSE;
361 	templates.list_modified = FALSE;
362 }
363 
prefs_template_window_setup(void)364 static void prefs_template_window_setup(void)
365 {
366 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
367 	GSList *tmpl_list;
368 	GSList *cur;
369 	gchar *title[1];
370 	gint row;
371 	Template *tmpl;
372 
373 	gtkut_box_set_reverse_order(GTK_BOX(templates.confirm_area),
374 				    !prefs_common.comply_gnome_hig);
375 	manage_window_set_transient(GTK_WINDOW(templates.window));
376 	gtk_widget_grab_focus(templates.ok_btn);
377 
378 	gtk_clist_freeze(clist);
379 	gtk_clist_clear(clist);
380 
381 	title[0] = _("(New)");
382 	row = gtk_clist_append(clist, title);
383 	gtk_clist_set_row_data(clist, row, NULL);
384 
385 	tmpl_list = template_read_config();
386 
387 	for (cur = tmpl_list; cur != NULL; cur = cur->next) {
388 		tmpl = (Template *)cur->data;
389 		title[0] = tmpl->name;
390 		row = gtk_clist_append(clist, title);
391 		gtk_clist_set_row_data(clist, row, tmpl);
392 	}
393 
394 	g_slist_free(tmpl_list);
395 
396 	gtk_clist_thaw(clist);
397 
398 	templates.entry_modified = FALSE;
399 	templates.list_modified = FALSE;
400 }
401 
prefs_template_clear(void)402 static void prefs_template_clear(void)
403 {
404 	Template *tmpl;
405 	gint row = 1;
406 
407 	while ((tmpl = gtk_clist_get_row_data
408 		(GTK_CLIST(templates.clist_tmpls), row)) != NULL) {
409 		template_free(tmpl);
410 		row++;
411 	}
412 
413 	gtk_clist_clear(GTK_CLIST(templates.clist_tmpls));
414 }
415 
prefs_template_deleted_cb(GtkWidget * widget,GdkEventAny * event,gpointer data)416 static gint prefs_template_deleted_cb(GtkWidget *widget, GdkEventAny *event,
417 				      gpointer data)
418 {
419 	prefs_template_cancel_cb();
420 	return TRUE;
421 }
422 
prefs_template_key_pressed_cb(GtkWidget * widget,GdkEventKey * event,gpointer data)423 static gboolean prefs_template_key_pressed_cb(GtkWidget *widget,
424 					      GdkEventKey *event, gpointer data)
425 {
426 	if (event && event->keyval == GDK_Escape)
427 		prefs_template_cancel_cb();
428 	return FALSE;
429 }
430 
prefs_template_changed_cb(GtkEditable * editable,gpointer data)431 static void prefs_template_changed_cb(GtkEditable *editable, gpointer data)
432 {
433 	templates.entry_modified = TRUE;
434 }
435 
prefs_template_ok_cb(void)436 static void prefs_template_ok_cb(void)
437 {
438 	GSList *tmpl_list;
439 
440 	if (templates.entry_modified) {
441 		if (alertpanel(_("Template is modified"),
442 			       _("Current modification is not applied. Finish without saving it?"),
443 			       GTK_STOCK_YES, GTK_STOCK_NO, NULL)
444 		    != G_ALERTDEFAULT)
445 			return;
446 	}
447 
448 	tmpl_list = prefs_template_get_list();
449 	template_set_config(tmpl_list);
450 	compose_reflect_prefs_all();
451 	gtk_clist_clear(GTK_CLIST(templates.clist_tmpls));
452 	gtk_widget_hide(templates.window);
453 	main_window_popup(main_window_get());
454 	inc_unlock();
455 }
456 
prefs_template_cancel_cb(void)457 static void prefs_template_cancel_cb(void)
458 {
459 	if (templates.entry_modified || templates.list_modified) {
460 		if (alertpanel(_("Templates are modified"),
461 			       _("Really discard modification to templates?"),
462 			       GTK_STOCK_YES, GTK_STOCK_NO, NULL)
463 		    != G_ALERTDEFAULT)
464 			return;
465 	}
466 
467 	prefs_template_clear();
468 	gtk_widget_hide(templates.window);
469 	main_window_popup(main_window_get());
470 	inc_unlock();
471 }
472 
prefs_template_select_cb(GtkCList * clist,gint row,gint column,GdkEvent * event)473 static void prefs_template_select_cb(GtkCList *clist, gint row, gint column,
474 				     GdkEvent *event)
475 {
476 	Template *tmpl;
477 	Template tmpl_def;
478 	GtkTextBuffer *buffer;
479 	GtkTextIter iter;
480 
481 	tmpl_def.name = _("Template");
482 	tmpl_def.subject = "";
483 	tmpl_def.to = "";
484 	tmpl_def.cc = "";
485 	tmpl_def.bcc = "";
486 	tmpl_def.replyto = "";
487 	tmpl_def.value = "";
488 
489 	if (!(tmpl = gtk_clist_get_row_data(clist, row)))
490 		tmpl = &tmpl_def;
491 
492 	gtk_entry_set_text(GTK_ENTRY(templates.entry_name), tmpl->name);
493 	gtk_entry_set_text(GTK_ENTRY(templates.entry_to),
494 			   tmpl->to ? tmpl->to : "");
495 	gtk_entry_set_text(GTK_ENTRY(templates.entry_cc),
496 			   tmpl->cc ? tmpl->cc : "");
497 	gtk_entry_set_text(GTK_ENTRY(templates.entry_bcc),
498 			   tmpl->bcc ? tmpl->bcc : "");
499 	gtk_entry_set_text(GTK_ENTRY(templates.entry_replyto),
500 			   tmpl->replyto ? tmpl->replyto : "");
501 	gtk_entry_set_text(GTK_ENTRY(templates.entry_subject),
502 			   tmpl->subject ? tmpl->subject : "");
503 
504 	buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(templates.text_value));
505 	gtk_text_buffer_set_text(buffer, "", 0);
506 	gtk_text_buffer_get_start_iter(buffer, &iter);
507 	gtk_text_buffer_insert(buffer, &iter, tmpl->value, -1);
508 
509 	templates.entry_modified = FALSE;
510 }
511 
prefs_template_get_list(void)512 static GSList *prefs_template_get_list(void)
513 {
514 	gint row = 1;
515 	GSList *tmpl_list = NULL;
516 	Template *tmpl;
517 
518 	while ((tmpl = gtk_clist_get_row_data
519 		(GTK_CLIST(templates.clist_tmpls), row)) != NULL) {
520 		tmpl->tmplid = row;
521 		tmpl_list = g_slist_append(tmpl_list, tmpl);
522 		row++;
523 	}
524 
525 	return tmpl_list;
526 }
527 
prefs_template_clist_set_row(gint row)528 static gint prefs_template_clist_set_row(gint row)
529 {
530 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
531 	Template *tmpl;
532 	Template *tmp_tmpl;
533 	GtkTextBuffer *buffer;
534 	GtkTextIter start, end;
535 	gchar *name;
536 	gchar *to;
537 	gchar *cc;
538 	gchar *bcc;
539 	gchar *replyto;
540 	gchar *subject;
541 	gchar *value;
542 	gchar *title[1];
543 
544 	g_return_val_if_fail(row != 0, -1);
545 
546 	buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(templates.text_value));
547 	gtk_text_buffer_get_start_iter(buffer, &start);
548 	gtk_text_buffer_get_end_iter(buffer, &end);
549 	value = gtk_text_buffer_get_text(buffer, &start, &end, FALSE);
550 
551 	if (value && *value != '\0') {
552 		gchar *parsed_buf;
553 		MsgInfo dummyinfo;
554 
555 		memset(&dummyinfo, 0, sizeof(MsgInfo));
556 		quote_fmt_init(&dummyinfo, NULL, NULL);
557 		quote_fmt_scan_string(value);
558 		quote_fmt_parse();
559 		parsed_buf = quote_fmt_get_buffer();
560 		if (!parsed_buf) {
561 			alertpanel_error(_("Template format error."));
562 			g_free(value);
563 			return -1;
564 		}
565 	}
566 
567 	name = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_name),
568 				      0, -1);
569 	to = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_to), 0, -1);
570 	cc = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_cc), 0, -1);
571 	bcc = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_bcc), 0, -1);
572 	replyto = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_replyto),
573 					 0, -1);
574 	subject = gtk_editable_get_chars(GTK_EDITABLE(templates.entry_subject),
575 					 0, -1);
576 
577 #define NULLIFY_IF_EMPTY(val)		\
578 	if (val && *val == '\0') {	\
579 		g_free(val);		\
580 		val = NULL;		\
581 	}
582 
583 	NULLIFY_IF_EMPTY(to);
584 	NULLIFY_IF_EMPTY(cc);
585 	NULLIFY_IF_EMPTY(bcc);
586 	NULLIFY_IF_EMPTY(replyto);
587 	NULLIFY_IF_EMPTY(subject);
588 
589 #undef NULLIFY_IF_EMPTY
590 
591 	tmpl = g_new(Template, 1);
592 	tmpl->name = name;
593 	tmpl->to = to;
594 	tmpl->cc = cc;
595 	tmpl->bcc = bcc;
596 	tmpl->replyto = replyto;
597 	tmpl->subject = subject;
598 	tmpl->value = value;
599 
600 	title[0] = name;
601 
602 	if (row < 0) {
603 		row = gtk_clist_append(clist, title);
604 	} else {
605 		gtk_clist_set_text(clist, row, 0, name);
606 		tmp_tmpl = gtk_clist_get_row_data(clist, row);
607 		if (tmp_tmpl)
608 			template_free(tmp_tmpl);
609 	}
610 
611 	gtk_clist_set_row_data(clist, row, tmpl);
612 	templates.entry_modified = FALSE;
613 	templates.list_modified = TRUE;
614 
615 	return row;
616 }
617 
prefs_template_register_cb(void)618 static void prefs_template_register_cb(void)
619 {
620 	prefs_template_clist_set_row(-1);
621 }
622 
prefs_template_substitute_cb(void)623 static void prefs_template_substitute_cb(void)
624 {
625 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
626 	Template *tmpl;
627 	gint row;
628 
629 	if (!clist->selection) return;
630 
631 	row = GPOINTER_TO_INT(clist->selection->data);
632 	if (row == 0) return;
633 
634 	tmpl = gtk_clist_get_row_data(clist, row);
635 	if (!tmpl) return;
636 
637 	prefs_template_clist_set_row(row);
638 }
639 
prefs_template_delete_cb(void)640 static void prefs_template_delete_cb(void)
641 {
642 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
643 	Template *tmpl;
644 	gint row;
645 
646 	if (!clist->selection) return;
647 	row = GPOINTER_TO_INT(clist->selection->data);
648 	if (row == 0) return;
649 
650 	if (alertpanel(_("Delete template"),
651 		       _("Do you really want to delete this template?"),
652 		       GTK_STOCK_YES, GTK_STOCK_NO, NULL) != G_ALERTDEFAULT)
653 		return;
654 
655 	tmpl = gtk_clist_get_row_data(clist, row);
656 	template_free(tmpl);
657 	gtk_clist_remove(clist, row);
658 	templates.list_modified = TRUE;
659 }
660 
prefs_template_up_cb(void)661 static void prefs_template_up_cb(void)
662 {
663 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
664 	gint row;
665 
666 	if (!clist->selection) return;
667 	row = GPOINTER_TO_INT(clist->selection->data);
668 	if (row > 1) {
669 		gtk_clist_row_move(clist, row, row - 1);
670 		templates.list_modified = TRUE;
671 	}
672 }
673 
prefs_template_down_cb(void)674 static void prefs_template_down_cb(void)
675 {
676 	GtkCList *clist = GTK_CLIST(templates.clist_tmpls);
677 	gint row;
678 
679 	if (!clist->selection) return;
680 	row = GPOINTER_TO_INT(clist->selection->data);
681 	if (row > 0 && row < clist->rows - 1) {
682 		gtk_clist_row_move(clist, row, row + 1);
683 		templates.list_modified = TRUE;
684 	}
685 }
686