Display length & check digit of entered EAN-13 in metadata dialogue
[ardour.git] / gtk2_ardour / session_metadata_dialog.cc
1 /*
2     Copyright (C) 2008 Paul Davis
3     Author: Sakari Bergen
4
5     This program is free software; you can redistribute it and/or modify it
6     under the terms of the GNU General Public License as published by the Free
7     Software Foundation; either version 2 of the License, or (at your option)
8     any later version.
9
10     This program is distributed in the hope that it will be useful, but WITHOUT
11     ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12     FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
13     for more details.
14
15     You should have received a copy of the GNU General Public License along
16     with this program; if not, write to the Free Software Foundation, Inc.,
17     675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20 #include "session_metadata_dialog.h"
21
22 #include <sstream>
23
24 #include <gtkmm2ext/utils.h>
25
26 #include "pbd/xml++.h"
27 #include "pbd/error.h"
28
29 #include "ardour/filename_extensions.h"
30 #include "ardour/session.h"
31 #include "ardour/session_utils.h"
32
33 #include "i18n.h"
34
35 using namespace std;
36 using namespace Glib;
37 using namespace PBD;
38
39 #define CALL_MEMBER_FN(object,ptrToMember)  ((object).*(ptrToMember))
40
41 /*** MetadataField ***/
42
43 MetadataField::MetadataField (string const & field_name) :
44   _name (field_name)
45 {
46 }
47
48 MetadataField::~MetadataField() { }
49
50 /* TextMetadataField */
51
52 TextMetadataField::TextMetadataField (Getter getter, Setter setter, string const & field_name, guint width ) :
53   MetadataField (field_name),
54   getter (getter),
55   setter (setter),
56   width (width)
57 {
58         entry = 0;
59         label = 0;
60         value_label = 0;
61 }
62
63 MetadataPtr
64 TextMetadataField::copy ()
65 {
66         return MetadataPtr (new TextMetadataField (getter, setter, _name, width));
67 }
68
69 void
70 TextMetadataField::save_data (ARDOUR::SessionMetadata & data) const
71 {
72         CALL_MEMBER_FN (data, setter) (_value);
73 }
74
75 void
76 TextMetadataField::load_data (ARDOUR::SessionMetadata const & data)
77 {
78         _value = CALL_MEMBER_FN (data, getter) ();
79         if (entry) {
80                 entry->set_text (_value);
81         }
82 }
83
84 Gtk::Widget &
85 TextMetadataField::name_widget ()
86 {
87         label = Gtk::manage (new Gtk::Label(_name + ':'));
88         label->set_alignment (1, 0.5);
89         return *label;
90 }
91
92 Gtk::Widget &
93 TextMetadataField::value_widget ()
94 {
95         value_label = Gtk::manage (new Gtk::Label(_value));
96         return *value_label;
97 }
98
99 Gtk::Widget &
100 TextMetadataField::edit_widget ()
101 {
102         entry = Gtk::manage (new Gtk::Entry());
103
104         entry->set_text (_value);
105         entry->set_width_chars (width);
106         entry->signal_changed().connect (sigc::mem_fun(*this, &TextMetadataField::update_value));
107
108         return *entry;
109 }
110
111 void
112 TextMetadataField::update_value ()
113 {
114         _value = entry->get_text ();
115 }
116
117 /* NumberMetadataField */
118
119 NumberMetadataField::NumberMetadataField (Getter getter, Setter setter, string const & field_name, guint numbers, guint width) :
120   MetadataField (field_name),
121   getter (getter),
122   setter (setter),
123   numbers (numbers),
124   width (width)
125 {
126         entry = 0;
127         label = 0;
128         value_label = 0;
129 }
130
131 MetadataPtr
132 NumberMetadataField::copy ()
133 {
134         return MetadataPtr (new NumberMetadataField (getter, setter, _name, numbers, width));
135 }
136
137 void
138 NumberMetadataField::save_data (ARDOUR::SessionMetadata & data) const
139 {
140         uint32_t number = str_to_uint (_value);
141         CALL_MEMBER_FN (data, setter) (number);
142 }
143
144 void
145 NumberMetadataField::load_data (ARDOUR::SessionMetadata const & data)
146 {
147         uint32_t number = CALL_MEMBER_FN (data, getter) ();
148         _value = uint_to_str (number);
149         if (entry) {
150                 entry->set_text (_value);
151         }
152 }
153
154 void
155 NumberMetadataField::update_value ()
156 {
157         // Accept only numbers that will fit into a uint32_t
158         uint32_t number = str_to_uint (entry->get_text());
159         _value = uint_to_str (number);
160         entry->set_text (_value);
161 }
162
163 Gtk::Widget &
164 NumberMetadataField::name_widget ()
165 {
166         label = Gtk::manage (new Gtk::Label(_name + ':'));
167         label->set_alignment (1, 0.5);
168         return *label;
169 }
170
171 Gtk::Widget &
172 NumberMetadataField::value_widget ()
173 {
174         value_label = Gtk::manage (new Gtk::Label(_value));
175         return *value_label;
176 }
177
178 Gtk::Widget &
179 NumberMetadataField::edit_widget ()
180 {
181         entry = Gtk::manage (new Gtk::Entry());
182
183         entry->set_text (_value);
184         entry->set_width_chars (width);
185         entry->set_max_length (numbers);
186         entry->signal_changed().connect (sigc::mem_fun(*this, &NumberMetadataField::update_value));
187
188         return *entry;
189 }
190
191 string
192 NumberMetadataField::uint_to_str (uint32_t i) const
193 {
194         std::ostringstream oss ("");
195         oss << i;
196         if (oss.str().compare("0")) {
197                 return oss.str();
198         } else {
199                 return "";
200         }
201 }
202
203 uint32_t
204 NumberMetadataField::str_to_uint (string const & str) const
205 {
206         string tmp (str);
207         string::size_type i;
208         while ((i = tmp.find_first_not_of("1234567890")) != string::npos) {
209                 tmp.erase (i, 1);
210         }
211
212         std::istringstream iss(tmp);
213         uint32_t result = 0;
214         iss >> result;
215         return result;
216 }
217
218
219
220 /* EAN13MetadataField */
221
222 EAN13MetadataField::EAN13MetadataField (Getter getter, Setter setter, string const & field_name, guint width) :
223   MetadataField (field_name),
224   getter (getter),
225   setter (setter),
226   width (width)
227 {
228         entry = 0;
229         label = 0;
230         value_label = 0;
231         status_label = Gtk::manage (new Gtk::Label (""));
232 }
233
234 MetadataPtr
235 EAN13MetadataField::copy ()
236 {
237         return MetadataPtr (new EAN13MetadataField (getter, setter, _name, width));
238 }
239
240 void
241 EAN13MetadataField::save_data (ARDOUR::SessionMetadata & data) const
242 {
243         CALL_MEMBER_FN (data, setter) (_value);
244 }
245
246 void
247 EAN13MetadataField::load_data (ARDOUR::SessionMetadata const & data)
248 {
249         _value = CALL_MEMBER_FN (data, getter) ();
250         if (entry) {
251                 entry->set_text (_value);
252         }
253         update_status ();
254 }
255
256 void
257 EAN13MetadataField::update_value ()
258 {
259         // Accept only numeric characters
260         _value = numeric_string (entry->get_text());
261         entry->set_text (_value);
262         update_status ();
263 }
264
265 void
266 EAN13MetadataField::update_status ()
267 {
268         int len = _value.length ();
269         if (len == 13) {
270                 // calculate EAN-13 modulo 10 check digit
271                 int sum = 0;
272                 const char *p = _value.c_str();
273                 for (int i =0; i < 12; i++) {
274                         char c = p[i] - '0';
275                         if (i % 2) {
276                                 sum += c;
277                         } else {
278                                 sum += c * 3;
279                         }
280                 }
281                 sum %= 10;
282                 if (sum == p[12] - '0') {
283                         status_label->set_markup (string_compose(
284                                                 "<span color=\"green\">%1: %2</span>",
285                                                 _("EAN Check digit OK"), sum));
286                 } else {
287                         status_label->set_markup (string_compose(
288                                                 "<span color=\"#ffa755\">%1: %2 (expected %3)</span>",
289                                                 _("EAN Check digit error"), p[12] - '0', sum));
290                 }
291         } else if (len > 0) {
292                 status_label->set_markup (string_compose(
293                                         "<span color=\"#ffa755\">%1: %2 (&lt;13)</span>",
294                                         _("EAN Length error"), len));
295         } else {
296                 status_label->set_text("");
297         }
298 }
299
300 Gtk::Widget &
301 EAN13MetadataField::name_widget ()
302 {
303         label = Gtk::manage (new Gtk::Label(_name + ':'));
304         label->set_alignment (1, 0.5);
305         return *label;
306 }
307
308 Gtk::Widget &
309 EAN13MetadataField::value_widget ()
310 {
311         value_label = Gtk::manage (new Gtk::Label(_value));
312         return *value_label;
313 }
314
315 Gtk::Widget &
316 EAN13MetadataField::edit_widget ()
317 {
318         entry = Gtk::manage (new Gtk::Entry());
319
320         entry->set_text (_value);
321         entry->set_width_chars (width);
322         entry->set_max_length (13);
323         entry->signal_changed().connect (sigc::mem_fun(*this, &EAN13MetadataField::update_value));
324
325         return *entry;
326 }
327
328 string
329 EAN13MetadataField::numeric_string (string const & str) const
330 {
331         string tmp (str);
332         string::size_type i;
333         while ((i = tmp.find_first_not_of("1234567890")) != string::npos) {
334                 tmp.erase (i, 1);
335         }
336         return tmp;
337 }
338
339 /* SessionMetadataSet */
340
341 SessionMetadataSet::SessionMetadataSet (string const & name)
342   : name (name)
343 {
344 }
345
346 void
347 SessionMetadataSet::add_data_field (MetadataPtr field)
348 {
349         list.push_back (field);
350 }
351
352 /* SessionMetadataSetEditable */
353
354 SessionMetadataSetEditable::SessionMetadataSetEditable (string const & name)
355   : SessionMetadataSet (name)
356 {
357         table.set_row_spacings (6);
358         table.set_col_spacings (12);
359         table.set_homogeneous (false);
360         vbox.pack_start (table, false, false);
361         vbox.set_spacing (6);
362         vbox.set_border_width (6);
363 }
364
365 Gtk::Widget &
366 SessionMetadataSetEditable::get_tab_widget ()
367 {
368         tab_widget.set_text (name);
369         return tab_widget;
370 }
371
372 void
373 SessionMetadataSetEditable::set_session (ARDOUR::Session * s)
374 {
375         SessionHandlePtr::set_session (s);
376
377         if (!_session) {
378                 return;
379         }
380
381         ARDOUR::SessionMetadata const & data = *(ARDOUR::SessionMetadata::Metadata());
382
383         table.resize (list.size(), 2);
384         uint32_t row = 0;
385         MetadataPtr field;
386         for (DataList::const_iterator it = list.begin(); it != list.end(); ++it) {
387                 field = *it;
388                 field->load_data (data);
389                 table.attach (field->name_widget(), 0, 1, row, row + 1, Gtk::FILL);
390                 table.attach (field->edit_widget(), 1, 2, row, row + 1);
391                 ++row;
392         }
393 }
394
395 void
396 SessionMetadataSetEditable::save_data ()
397 {
398         ARDOUR::SessionMetadata & data = *(ARDOUR::SessionMetadata::Metadata());
399         for (DataList::const_iterator it = list.begin(); it != list.end(); ++it) {
400                 (*it)->save_data(data);
401         }
402 }
403
404 /* SessionMetadataSetImportable */
405
406 SessionMetadataSetImportable::SessionMetadataSetImportable (string const & name)
407   : SessionMetadataSet (name)
408   , session_list (list)
409 {
410         tree = Gtk::ListStore::create (tree_cols);
411         tree_view.set_model (tree);
412
413         Gtk::TreeView::Column * viewcol;
414
415         // Add import column
416         Gtk::CellRendererToggle * import_render = Gtk::manage(new Gtk::CellRendererToggle());
417         import_render->signal_toggled().connect (sigc::mem_fun(*this, &SessionMetadataSetImportable::selection_changed));
418         viewcol = Gtk::manage(new Gtk::TreeView::Column (_("Import"), *import_render));
419         viewcol->add_attribute (import_render->property_active(), tree_cols.import);
420         tree_view.append_column (*viewcol);
421
422         // Add field name column
423         tree_view.append_column(_("Field"), tree_cols.field);
424
425         // Add values column with pango markup
426         Gtk::CellRendererText * values_render = Gtk::manage(new Gtk::CellRendererText());
427         viewcol = Gtk::manage(new Gtk::TreeView::Column (_("Values (current value on top)"), *values_render));
428         viewcol->add_attribute (values_render->property_markup(), tree_cols.values);
429         tree_view.append_column (*viewcol);
430
431         select_all_check.signal_toggled().connect (sigc::mem_fun(*this, &SessionMetadataSetImportable::select_all));
432 }
433
434 Gtk::Widget &
435 SessionMetadataSetImportable::get_tab_widget ()
436 {
437         tab_widget.set_text (name);
438         return tab_widget;
439 }
440
441 Gtk::Widget &
442 SessionMetadataSetImportable::get_select_all_widget ()
443 {
444         select_all_check.set_label (name);
445         return select_all_check;
446 }
447
448 void
449 SessionMetadataSetImportable::load_extra_data (ARDOUR::SessionMetadata const & data)
450 {
451         if (!_session) {
452                 error << string_compose (_("programming error: %1"), "no session set for SessionMetaDataSetImportable (in load_data)!") << endmsg;
453                 return;
454         }
455
456         ARDOUR::SessionMetadata const & session_data = *(ARDOUR::SessionMetadata::Metadata());
457
458         MetadataPtr session_field;
459         MetadataPtr import_field;
460         DataList::iterator session_it;
461         DataList::iterator import_it;
462
463         // Copy list and load data to import
464         for (session_it = session_list.begin(); session_it != session_list.end(); ++session_it) {
465                 session_field = *session_it;
466                 session_field->load_data(session_data);
467                 import_list.push_back (session_field->copy());
468         }
469
470         // Fill widget
471         session_it = session_list.begin();
472         import_it = import_list.begin();
473         while (session_it != session_list.end() && import_it != import_list.end()) { // _should_ be the same...
474                 session_field = *session_it;
475                 import_field = *import_it;
476
477                 import_field->load_data(data); // hasn't been done yet
478
479                 // Make string for values TODO get color from somewhere?
480                 string values = "<span weight=\"ultralight\" color=\"#777\">" + session_field->value() + "</span>\n"
481                         + "<span weight=\"bold\">" + import_field->value() + "</span>";
482
483                 Gtk::TreeModel::iterator row_iter = tree->append();
484                 Gtk::TreeModel::Row row = *row_iter;
485
486                 row[tree_cols.field] = import_field->name();
487                 row[tree_cols.values] = values;
488                 row[tree_cols.import] = false;
489                 row[tree_cols.data] = import_field;
490
491                 ++session_it;
492                 ++import_it;
493         }
494 }
495
496 void
497 SessionMetadataSetImportable::save_data ()
498 {
499         if (!_session) {
500                 error << string_compose (_("programming error: %1"), "no session set for SessionMetaDataSetImportable (in import_data)!") << endmsg;
501                 return;
502         }
503
504         ARDOUR::SessionMetadata & session_data = *(ARDOUR::SessionMetadata::Metadata());
505
506         Gtk::TreeModel::Children fields = tree->children();
507         Gtk::TreeModel::Children::iterator it;
508         for (it = fields.begin(); it != fields.end(); ++it) {
509                 if ((*it)[tree_cols.import]) {
510                         MetadataPtr field = (*it)[tree_cols.data];
511                         field->save_data (session_data);
512                 }
513         }
514 }
515
516 void
517 SessionMetadataSetImportable::select_all ()
518 {
519         select_all_check.set_inconsistent (false);
520         bool state = select_all_check.get_active();
521
522         Gtk::TreeModel::Children fields = tree->children();
523         Gtk::TreeModel::Children::iterator it;
524         for (it = fields.begin(); it != fields.end(); ++it) {
525                 (*it)[tree_cols.import] = state;
526         }
527 }
528
529 void
530 SessionMetadataSetImportable::selection_changed (string const & path)
531 {
532         select_all_check.set_inconsistent (true);
533
534         Gtk::TreeModel::iterator iter = tree->get_iter (path);
535         bool value((*iter)[tree_cols.import]);
536         (*iter)[tree_cols.import] = !value;
537 }
538
539 /* SessionMetadataDialog */
540
541 template <typename DataSet>
542 SessionMetadataDialog<DataSet>::SessionMetadataDialog (string const & name) :
543   ArdourDialog (name, true)
544 {
545         cancel_button = add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
546         cancel_button->signal_clicked().connect (sigc::mem_fun(*this, &SessionMetadataDialog::end_dialog));
547         save_button = add_button (Gtk::Stock::OK, Gtk::RESPONSE_ACCEPT);
548         save_button->signal_clicked().connect (sigc::mem_fun(*this, &SessionMetadataDialog::save_and_close));
549 }
550
551 template <typename DataSet>
552 void
553 SessionMetadataDialog<DataSet>::init_data ( bool skip_user )
554 {
555         if (!_session) {
556                 error << string_compose (_("programming error: %1"), "no session set for SessionMetaDataDialog (in init_data)!") << endmsg;
557                 return;
558         }
559
560         if (!skip_user)
561                 init_user_data ();
562         init_track_data ();
563         init_album_data ();
564         init_people_data ();
565         init_school_data ();
566
567         for (DataSetList::iterator it = data_list.begin(); it != data_list.end(); ++it) {
568                 (*it)->set_session (_session);
569
570                 notebook.append_page ((*it)->get_widget(), (*it)->get_tab_widget());
571         }
572 }
573
574 template <typename DataSet>
575 void
576 SessionMetadataDialog<DataSet>::load_extra_data (ARDOUR::SessionMetadata const & data)
577 {
578         for (DataSetList::iterator it = data_list.begin(); it != data_list.end(); ++it) {
579                 (*it)->load_extra_data (data);
580         }
581 }
582
583 template <typename DataSet>
584 void
585 SessionMetadataDialog<DataSet>::save_data ()
586 {
587         for (DataSetList::iterator it = data_list.begin(); it != data_list.end(); ++it) {
588                 (*it)->save_data ();
589         }
590 }
591
592 template <typename DataSet>
593 void
594 SessionMetadataDialog<DataSet>::save_and_close ()
595 {
596         save_data ();
597         _session->set_dirty();
598         end_dialog ();
599 }
600
601 template <typename DataSet>
602 void
603 SessionMetadataDialog<DataSet>::end_dialog ()
604 {
605         hide_all();
606 }
607
608 template <typename DataSet>
609 void
610 SessionMetadataDialog<DataSet>::warn_user (string const & string)
611 {
612         Gtk::MessageDialog msg (string, false, Gtk::MESSAGE_WARNING, Gtk::BUTTONS_OK, true);
613         msg.run();
614 }
615
616 template <typename DataSet>
617 boost::shared_ptr<std::list<Gtk::Widget *> >
618 SessionMetadataDialog<DataSet>::get_custom_widgets (WidgetFunc f)
619 {
620         WidgetListPtr list (new WidgetList);
621         for (DataSetList::iterator it = data_list.begin(); it != data_list.end(); ++it)
622         {
623                 DataSet * set = dynamic_cast<DataSet *> (it->get());
624                 list->push_back (& CALL_MEMBER_FN (*set, f) ());
625         }
626
627         return list;
628 }
629
630 template <typename DataSet>
631 void
632 SessionMetadataDialog<DataSet>::add_widget (Gtk::Widget & widget)
633 {
634         get_vbox()->pack_start (widget, true, true, 0);
635 }
636
637 template <typename DataSet>
638 void
639 SessionMetadataDialog<DataSet>::init_user_data ()
640 {
641         DataSetPtr data_set (new DataSet (_("User")));
642         data_list.push_back (data_set);
643
644         MetadataPtr ptr;
645
646         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::user_name, &ARDOUR::SessionMetadata::set_user_name, _("Name")));
647         data_set->add_data_field (ptr);
648
649         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::user_email, &ARDOUR::SessionMetadata::set_user_email, _("Email")));
650         data_set->add_data_field (ptr);
651
652         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::user_web, &ARDOUR::SessionMetadata::set_user_web, _("Web")));
653         data_set->add_data_field (ptr);
654
655         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::organization, &ARDOUR::SessionMetadata::set_organization, _("Organization")));
656         data_set->add_data_field (ptr);
657
658         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::country, &ARDOUR::SessionMetadata::set_country, _("Country")));
659         data_set->add_data_field (ptr);
660
661 }
662
663 template <typename DataSet>
664 void
665 SessionMetadataDialog<DataSet>::init_track_data ()
666 {
667         DataSetPtr data_set (new DataSet (_("Track")));
668         data_list.push_back (data_set);
669
670         MetadataPtr ptr;
671
672         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::title, &ARDOUR::SessionMetadata::set_title, _("Title")));
673         data_set->add_data_field (ptr);
674
675         ptr = MetadataPtr (new NumberMetadataField (&ARDOUR::SessionMetadata::track_number, &ARDOUR::SessionMetadata::set_track_number, _("Track Number"), 3));
676         data_set->add_data_field (ptr);
677
678         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::subtitle, &ARDOUR::SessionMetadata::set_subtitle, _("Subtitle")));
679         data_set->add_data_field (ptr);
680
681         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::grouping, &ARDOUR::SessionMetadata::set_grouping, _("Grouping")));
682         data_set->add_data_field (ptr);
683
684         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::artist, &ARDOUR::SessionMetadata::set_artist, _("Artist")));
685         data_set->add_data_field (ptr);
686
687         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::genre, &ARDOUR::SessionMetadata::set_genre, _("Genre")));
688         data_set->add_data_field (ptr);
689
690         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::comment, &ARDOUR::SessionMetadata::set_comment, _("Comment")));
691         data_set->add_data_field (ptr);
692
693         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::copyright, &ARDOUR::SessionMetadata::set_copyright, _("Copyright")));
694         data_set->add_data_field (ptr);
695 }
696
697 template <typename DataSet>
698 void
699 SessionMetadataDialog<DataSet>::init_album_data ()
700 {
701         DataSetPtr data_set (new DataSet (_("Album")));
702         data_list.push_back (data_set);
703
704         MetadataPtr ptr;
705
706         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::album, &ARDOUR::SessionMetadata::set_album, _("Album")));
707         data_set->add_data_field (ptr);
708
709         ptr = MetadataPtr (new NumberMetadataField (&ARDOUR::SessionMetadata::year, &ARDOUR::SessionMetadata::set_year, _("Year"), 4));
710         data_set->add_data_field (ptr);
711
712         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::album_artist, &ARDOUR::SessionMetadata::set_album_artist, _("Album Artist")));
713         data_set->add_data_field (ptr);
714
715         ptr = MetadataPtr (new NumberMetadataField (&ARDOUR::SessionMetadata::total_tracks, &ARDOUR::SessionMetadata::set_total_tracks, _("Total Tracks"), 3));
716         data_set->add_data_field (ptr);
717
718         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::disc_subtitle, &ARDOUR::SessionMetadata::set_disc_subtitle, _("Disc Subtitle")));
719         data_set->add_data_field (ptr);
720
721         ptr = MetadataPtr (new NumberMetadataField (&ARDOUR::SessionMetadata::disc_number, &ARDOUR::SessionMetadata::set_disc_number, _("Disc Number"), 2));
722         data_set->add_data_field (ptr);
723
724         ptr = MetadataPtr (new NumberMetadataField (&ARDOUR::SessionMetadata::total_discs, &ARDOUR::SessionMetadata::set_total_discs, _("Total Discs"), 2));
725         data_set->add_data_field (ptr);
726
727         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::compilation, &ARDOUR::SessionMetadata::set_compilation, _("Compilation")));
728         data_set->add_data_field (ptr);
729
730         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::isrc, &ARDOUR::SessionMetadata::set_isrc, _("ISRC")));
731         data_set->add_data_field (ptr);
732
733         ptr = MetadataPtr (new EAN13MetadataField (&ARDOUR::SessionMetadata::barcode, &ARDOUR::SessionMetadata::set_barcode, _("EAN barcode")));
734         data_set->add_data_field (ptr);
735
736         // EAN13MetadataField is the only kind of MetadataField which has a status label.
737         EAN13MetadataField &emf = (EAN13MetadataField &) *ptr;
738         ((Gtk::VBox &) data_set->get_widget()).pack_end (*emf.status_label);
739         emf.update_status ();
740 }
741
742 template <typename DataSet>
743 void
744 SessionMetadataDialog<DataSet>::init_people_data ()
745 {
746         DataSetPtr data_set (new DataSet (_("People")));
747         data_list.push_back (data_set);
748
749         MetadataPtr ptr;
750
751         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::lyricist, &ARDOUR::SessionMetadata::set_lyricist, _("Lyricist")));
752         data_set->add_data_field (ptr);
753
754         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::composer, &ARDOUR::SessionMetadata::set_composer, _("Composer")));
755         data_set->add_data_field (ptr);
756
757         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::conductor, &ARDOUR::SessionMetadata::set_conductor, _("Conductor")));
758         data_set->add_data_field (ptr);
759
760         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::remixer, &ARDOUR::SessionMetadata::set_remixer, _("Remixer")));
761         data_set->add_data_field (ptr);
762
763         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::arranger, &ARDOUR::SessionMetadata::set_arranger, _("Arranger")));
764         data_set->add_data_field (ptr);
765
766         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::engineer, &ARDOUR::SessionMetadata::set_engineer, _("Engineer")));
767         data_set->add_data_field (ptr);
768
769         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::producer, &ARDOUR::SessionMetadata::set_producer, _("Producer")));
770         data_set->add_data_field (ptr);
771
772         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::dj_mixer, &ARDOUR::SessionMetadata::set_dj_mixer, _("DJ Mixer")));
773         data_set->add_data_field (ptr);
774
775         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::mixer, &ARDOUR::SessionMetadata::set_mixer, S_("Metadata|Mixer")));
776         data_set->add_data_field (ptr);
777 }
778
779 template <typename DataSet>
780 void
781 SessionMetadataDialog<DataSet>::init_school_data ()
782 {
783         DataSetPtr data_set (new DataSet (_("School")));
784         data_list.push_back (data_set);
785
786         MetadataPtr ptr;
787
788         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::instructor, &ARDOUR::SessionMetadata::set_instructor, _("Instructor")));
789         data_set->add_data_field (ptr);
790
791         ptr = MetadataPtr (new TextMetadataField (&ARDOUR::SessionMetadata::course, &ARDOUR::SessionMetadata::set_course, _("Course")));
792         data_set->add_data_field (ptr);
793
794 }
795
796 /* SessionMetadataEditor */
797
798 SessionMetadataEditor::SessionMetadataEditor () :
799   SessionMetadataDialog<SessionMetadataSetEditable> (_("Edit Session Metadata"))
800 {
801
802 }
803
804 SessionMetadataEditor::~SessionMetadataEditor ()
805 {
806         // Remove pages from notebook to get rid of gsignal runtime warnings
807         notebook.pages().clear();
808 }
809
810 void
811 SessionMetadataEditor::run ()
812 {
813         init_data ();
814         init_gui();
815
816         ArdourDialog::run();
817 }
818
819 void
820 SessionMetadataEditor::init_gui ()
821 {
822         add_widget (notebook);
823
824         show_all();
825 }
826
827 /* SessionMetadataImporter */
828
829 SessionMetadataImporter::SessionMetadataImporter () :
830   SessionMetadataDialog<SessionMetadataSetImportable> (_("Import session metadata"))
831 {
832
833 }
834
835 SessionMetadataImporter::~SessionMetadataImporter ()
836 {
837         // Remove pages from notebook to get rid of gsignal runtime warnings
838         notebook.pages().clear();
839 }
840
841 void
842 SessionMetadataImporter::run ()
843 {
844         if (!_session) {
845                 error << string_compose (_("programming error: %1"), "no session set for SessionMetaDataImporter (in run)!") << endmsg;
846                 return;
847         }
848
849         /* Open session file selector */
850
851         Gtk::FileChooserDialog session_selector(_("Choose session to import metadata from"), Gtk::FILE_CHOOSER_ACTION_OPEN);
852         session_selector.add_button (Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
853         session_selector.add_button (Gtk::Stock::OPEN, Gtk::RESPONSE_ACCEPT);
854         session_selector.set_default_response(Gtk::RESPONSE_ACCEPT);
855
856         Gtk::FileFilter session_filter;
857         session_filter.add_pattern (string_compose(X_("*%1"), ARDOUR::statefile_suffix));
858         session_filter.set_name (string_compose (_("%1 sessions"), PROGRAM_NAME));
859         session_selector.add_filter (session_filter);
860         session_selector.set_filter (session_filter);
861
862         int response = session_selector.run();
863         session_selector.hide ();
864
865         switch (response) {
866         case Gtk::RESPONSE_ACCEPT:
867                 break;
868         default:
869                 return;
870         }
871
872         string session_path = session_selector.get_filename();
873         string path, name;
874         bool isnew;
875
876         if (session_path.length() > 0) {
877                 if (ARDOUR::find_session (session_path, path, name, isnew) != 0) {
878                         return;
879                 }
880         } else {
881                 return;
882         }
883
884         /* We have a session: load the data and run dialog */
885
886         string filename = Glib::build_filename (path, name + ARDOUR::statefile_suffix);
887         XMLTree session_tree;
888         if (!session_tree.read (filename)) {
889                 warn_user (_("This session file could not be read!"));
890                 return;
891         }
892
893         /* XXX GET VERSION FROM TREE */
894         int version = 3000;
895
896         XMLNode * node = session_tree.root()->child ("Metadata");
897
898         if (!node) {
899                 warn_user (_("The session file didn't contain metadata!\nMaybe this is an old session format?"));
900                 return;
901         }
902
903         //create a temporary 
904         ARDOUR::SessionMetadata data;
905         data.set_state (*node, version);
906         init_data ( true );  //skip user data here
907         load_extra_data (data);
908         init_gui();
909
910         ArdourDialog::run();
911 }
912
913 void
914 SessionMetadataImporter::init_gui ()
915 {
916         // Select all from -widget
917         add_widget (selection_hbox);
918         selection_label.set_text (_("Import all from:"));
919         selection_hbox.pack_start (selection_label, false, false);
920
921         WidgetListPtr list = get_custom_widgets (&SessionMetadataSetImportable::get_select_all_widget);
922         for (WidgetList::iterator it = list->begin(); it != list->end(); ++it) {
923                 selection_hbox.pack_start (**it, false, false, 6);
924         }
925
926         add_widget (notebook);
927
928         show_all();
929 }