de6af96f0463af4b011f9741f999a15eeec20723
[ardour.git] / gtk2_ardour / video_server_dialog.cc
1 /*
2     Copyright (C) 2010 Paul Davis
3     Author: Robin Gareus <robin@gareus.org>
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18
19 */
20 #include <cstdio>
21 #include <cmath>
22
23 #include <sigc++/bind.h>
24
25 #include "pbd/error.h"
26 #include "pbd/file_utils.h"
27 #include "ardour/session_directory.h"
28 #include "gtkmm2ext/utils.h"
29 #include "ardour/template_utils.h"
30 #include "ardour/session.h"
31
32 #ifdef PLATFORM_WINDOWS
33 #include <windows.h>
34 #include <shlobj.h> // CSIDL_*
35 #include "pbd/windows_special_dirs.h"
36 #endif
37
38 #ifdef interface
39 #undef interface
40 #endif
41
42 #include "video_server_dialog.h"
43 #include "utils_videotl.h"
44 #include "i18n.h"
45
46 #ifdef PLATFORM_WINDOWS
47 #include <windows.h>
48 #endif
49
50 using namespace Gtk;
51 using namespace std;
52 using namespace PBD;
53 using namespace ARDOUR;
54 using namespace VideoUtils;
55
56 VideoServerDialog::VideoServerDialog (Session* s)
57         : ArdourDialog (_("Launch Video Server"))
58         , path_label (_("Server Executable:"), Gtk::ALIGN_LEFT)
59         , path_browse_button (_("Browse"))
60         , docroot_label (_("Server Docroot:"), Gtk::ALIGN_LEFT)
61         , docroot_browse_button (_("Browse"))
62         , listenport_adjustment (1554, 1025, 65536, 1, 10, 0)
63         , listenport_spinner (listenport_adjustment)
64         , cachesize_adjustment (256, 32, 32768, 1, 32, 0)
65         , cachesize_spinner (cachesize_adjustment)
66         , showagain_checkbox (_("Don't show this dialog again. (Reset in Edit->Preferences)."))
67 {
68         set_session (s);
69
70         set_name ("VideoServerDialog");
71         set_modal (true);
72         set_skip_taskbar_hint (true);
73         set_resizable (false);
74
75         Gtk::Label* l;
76         VBox* vbox = manage (new VBox);
77         VBox* options_box = manage (new VBox);
78         HBox* path_hbox = manage (new HBox);
79         HBox* docroot_hbox = manage (new HBox);
80
81         path_entry.set_width_chars(38);
82         path_browse_button.set_name ("PaddedButton");
83         path_entry.set_text("/usr/bin/harvid");
84         docroot_entry.set_width_chars(38);
85         docroot_entry.set_text(Config->get_video_server_docroot());
86         docroot_browse_button.set_name ("PaddedButton");
87
88         listenaddr_combo.set_name ("PaddedButton");
89 #ifndef __APPLE__
90         /* Note: on OSX icsd is not able to bind to IPv4 localhost */
91         listenaddr_combo.append_text("127.0.0.1");
92 #endif
93         listenaddr_combo.append_text("0.0.0.0");
94         listenaddr_combo.set_active(0);
95
96 #ifdef PLATFORM_WINDOWS
97         HKEY key;
98         DWORD size = PATH_MAX;
99         char tmp[PATH_MAX+1];
100         const char *program_files = PBD::get_win_special_folder (CSIDL_PROGRAM_FILES);
101 #endif
102
103         std::string icsd_file_path;
104         if (find_file (PBD::Searchpath(Glib::getenv("PATH")), X_("harvid"), icsd_file_path)) {
105                 path_entry.set_text(icsd_file_path);
106         }
107 #ifdef PLATFORM_WINDOWS
108         else if ( (ERROR_SUCCESS == RegOpenKeyExA (HKEY_LOCAL_MACHINE, "Software\\RSS\\harvid", 0, KEY_READ, &key))
109                         &&  (ERROR_SUCCESS == RegQueryValueExA (key, "Install_Dir", 0, NULL, reinterpret_cast<LPBYTE>(tmp), &size))
110                         )
111         {
112                 path_entry.set_text(g_build_filename(Glib::locale_to_utf8(tmp).c_str(), "harvid.exe", 0));
113         }
114         else if (program_files && Glib::file_test(g_build_filename(program_files, "harvid", "harvid.exe", 0), Glib::FILE_TEST_EXISTS))
115         {
116                 path_entry.set_text(g_build_filename(program_files, "harvid", "harvid.exe", 0));
117         }
118 #endif
119         /* generic fallbacks to try */
120         else if (Glib::file_test(X_("C:\\Program Files\\harvid\\harvid.exe"), Glib::FILE_TEST_EXISTS)) {
121                 path_entry.set_text(X_("C:\\Program Files\\harvid\\harvid.exe"));
122         }
123         else {
124                 PBD::warning <<
125                         string_compose(
126                                         _("The external video server 'harvid' can not be found.\n"
127                                                 "The tool is included with the %1 releases from ardour.org, "
128                                                 "alternatively you can download it from http://x42.github.com/harvid/ "
129                                                 "or acquire it from your distribution.\n"
130                                                 "\n"
131                                                 "see also http://manual.ardour.org/video-timeline/setup/"
132                                          ), PROGRAM_NAME)
133                         << endmsg;
134         }
135
136
137         if (docroot_entry.get_text().empty()) {
138           std::string docroot =  Glib::path_get_dirname(_session->session_directory().root_path());
139           if ((docroot.empty() || docroot.at(docroot.length()-1) != '/')) { docroot += "/"; }
140                 docroot_entry.set_text(docroot);
141         }
142
143         path_hbox->pack_start (path_label, false, false, 3);
144         path_hbox->pack_start (path_entry, true, true, 3);
145         path_hbox->pack_start (path_browse_button, false, false, 3);
146
147         docroot_hbox->pack_start (docroot_label, false, false, 3);
148         docroot_hbox->pack_start (docroot_entry, true, true, 3);
149         docroot_hbox->pack_start (docroot_browse_button, false, false, 3);
150
151         l = manage (new Label (_("<b>Options</b>"), Gtk::ALIGN_LEFT, Gtk::ALIGN_CENTER, false));
152         l->set_use_markup ();
153         options_box->pack_start (*l, false, true, 4);
154
155         Table* t = manage (new Table (2, 3));
156         t->set_spacings (4);
157         options_box->pack_start (*t, true, true, 4);
158
159         l = manage (new Label (_("Listen Address:")));
160         l->set_alignment (0, 0.5);
161         t->attach (*l, 0, 1, 0, 1, FILL);
162         t->attach (listenaddr_combo, 1, 2, 0, 1);
163
164         l = manage (new Label (_("Listen Port:")));
165         l->set_alignment (0, 0.5);
166         t->attach (*l, 0, 1, 1, 2, FILL);
167         t->attach (listenport_spinner, 1, 2, 1, 2);
168
169         l = manage (new Label (_("Cache Size:")));
170         l->set_alignment (0, 0.5);
171         t->attach (*l, 0, 1, 2, 3, FILL);
172         t->attach (cachesize_spinner, 1, 2, 2, 3);
173
174         l = manage (new Label (string_compose(
175                                         _("%1 relies on an external video server for the videotimeline.\nThe server configured in Edit -> Preferences -> Video is not reachable.\nDo you want %1 to launch 'harvid' on this machine?"), PROGRAM_NAME)
176                                 , Gtk::ALIGN_LEFT, Gtk::ALIGN_CENTER, false));
177         l->set_max_width_chars(80);
178         l->set_line_wrap();
179         vbox->pack_start (*l, true, true, 4);
180         vbox->pack_start (*path_hbox, false, false);
181         if (Config->get_video_advanced_setup()){
182                 vbox->pack_start (*docroot_hbox, false, false);
183         } else {
184                 docroot_entry.set_text(X_("/"));
185                 listenport_spinner.set_sensitive(false);
186         }
187         vbox->pack_start (*options_box, false, true);
188
189         get_vbox()->set_spacing (4);
190         get_vbox()->pack_start (*vbox, false, false);
191         get_vbox()->pack_start (showagain_checkbox, false, false);
192         showagain_checkbox.set_active(!Config->get_show_video_server_dialog());
193
194         path_browse_button.signal_clicked().connect (sigc::mem_fun (*this, &VideoServerDialog::open_path_dialog));
195         docroot_browse_button.signal_clicked().connect (sigc::mem_fun (*this, &VideoServerDialog::open_docroot_dialog));
196
197         show_all_children ();
198         add_button (Stock::CANCEL, RESPONSE_CANCEL);
199         add_button (Stock::EXECUTE, RESPONSE_ACCEPT);
200 }
201
202 VideoServerDialog::~VideoServerDialog ()
203 {
204 }
205
206 void
207 VideoServerDialog::on_show ()
208 {
209         Dialog::on_show ();
210 }
211
212 void
213 VideoServerDialog::open_path_dialog ()
214 {
215         Gtk::FileChooserDialog dialog(_("Set Video Server Executable"), Gtk::FILE_CHOOSER_ACTION_OPEN);
216         dialog.set_filename (path_entry.get_text());
217
218         dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
219         dialog.add_button(Gtk::Stock::OK, Gtk::RESPONSE_OK);
220
221         int result = dialog.run();
222
223         if (result == Gtk::RESPONSE_OK) {
224                 std::string filename = dialog.get_filename();
225
226                 if (filename.length()) {
227                         path_entry.set_text (filename);
228                 }
229         }
230 }
231
232 void
233 VideoServerDialog::open_docroot_dialog ()
234 {
235         Gtk::FileChooserDialog dialog(_("Server docroot"), Gtk::FILE_CHOOSER_ACTION_SELECT_FOLDER);
236         dialog.set_filename (docroot_entry.get_text());
237
238         dialog.add_button(Gtk::Stock::CANCEL, Gtk::RESPONSE_CANCEL);
239         dialog.add_button(Gtk::Stock::OK, Gtk::RESPONSE_OK);
240
241         int result = dialog.run();
242
243         if (result == Gtk::RESPONSE_OK) {
244                 std::string dirname = dialog.get_filename();
245
246                 if (dirname.empty() || dirname.at(dirname.length()-1) != G_DIR_SEPARATOR) {
247                         dirname += "/";
248                 }
249
250                 if (dirname.length()) {
251                         docroot_entry.set_text (dirname);
252                 }
253         }
254 }
255
256 std::string
257 VideoServerDialog::get_docroot () {
258         return docroot_entry.get_text();
259 }